@mindline/sync 1.0.105 → 1.0.107
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.vs/slnx.sqlite +0 -0
- package/.vs/sync/CopilotIndices/17.14.260.54502/CodeChunks.db +0 -0
- package/.vs/sync/CopilotIndices/17.14.260.54502/SemanticSymbols.db +0 -0
- package/.vs/sync/FileContentIndex/0a490c4d-e4ea-4e8d-8989-720009d15f9d.vsidx +0 -0
- package/.vs/sync/FileContentIndex/173d280f-33b3-451d-8054-08fc6c991498.vsidx +0 -0
- package/.vs/sync/FileContentIndex/e27fae98-55a0-4376-b1eb-e2efa630b10b.vsidx +0 -0
- package/.vs/sync/FileContentIndex/fae8821f-9f13-46f5-91ef-05b837e2f35a.vsidx +0 -0
- package/.vs/sync/v17/.wsuo +0 -0
- package/.vs/sync/v17/DocumentLayout.backup.json +86 -28
- package/.vs/sync/v17/DocumentLayout.json +89 -29
- package/.vs/sync.slnx/FileContentIndex/13d6a28d-932f-4eb6-818c-efea35f24838.vsidx +0 -0
- package/.vs/sync.slnx/FileContentIndex/46af7fa0-1a65-4634-9faa-600623e92173.vsidx +0 -0
- package/.vs/sync.slnx/FileContentIndex/75bd0095-9fc3-4f35-90e9-f4022b8e4b55.vsidx +0 -0
- package/.vs/sync.slnx/FileContentIndex/92b75297-fe76-448d-80d4-cbf5ef4fcd60.vsidx +0 -0
- package/.vs/sync.slnx/FileContentIndex/eb2f80ef-22e0-4cc0-a8fb-f7837f87af2d.vsidx +0 -0
- package/.vs/sync.slnx/config/applicationhost.config +1011 -0
- package/.vs/sync.slnx/v18/.wsuo +0 -0
- package/.vs/sync.slnx/v18/DocumentLayout.backup.json +104 -0
- package/.vs/sync.slnx/v18/DocumentLayout.json +87 -0
- package/dist/src/index.d.ts +8 -4
- package/dist/sync.es.js +985 -931
- package/dist/sync.es.js.map +1 -1
- package/dist/sync.umd.js +24 -24
- package/dist/sync.umd.js.map +1 -1
- package/package.json +1 -1
- package/src/index.ts +215 -55
- package/.vs/sync/FileContentIndex/069feaec-de6f-456e-9a9f-3a7edff9927d.vsidx +0 -0
- package/.vs/sync/FileContentIndex/3b17981b-603f-4e34-8b49-f18ef10f5f91.vsidx +0 -0
- package/.vs/sync/FileContentIndex/5c93d870-257a-4d35-93a3-48d299fa5b1a.vsidx +0 -0
- package/.vs/sync/FileContentIndex/cce6d665-d5bc-4e83-b1ff-dabf19413432.vsidx +0 -0
- /package/.vs/sync/FileContentIndex/{db560492-d007-4291-a6b3-50ccdcbd66b4.vsidx → 72fe2c03-a5ac-417e-b6be-5d2e18aa7d59.vsidx} +0 -0
package/dist/sync.es.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"sync.es.js","sources":["../src/index.ts"],"sourcesContent":["//index.ts - published interface - AAD implementations, facade to Mindline SyncConfig API\r\nimport * as signalR from \"@microsoft/signalr\"\r\nimport { AccountInfo } from \"@azure/msal-common\";\r\nimport { IPublicClientApplication, AuthenticationResult } from \"@azure/msal-browser\"\r\nimport { deserializeArray } from 'class-transformer';\r\nimport users from \"./users.json\";\r\nimport tenants from \"./tenants.json\";\r\nimport configs from \"./configs.json\";\r\nimport workspaces from \"./workspaces.json\";\r\nimport syncmilestones from './syncmilestones.json';\r\nimport resources from './resources.json';\r\nimport actors from './actors.json';\r\n\r\nconst FILTER_FIELD = \"workspaceIDs\";\r\n// called by unit tests\r\nexport function sum(a: number, b: number): number {\r\n return a + b;\r\n}\r\nexport function helloNpm(): string {\r\n return \"hello NPM\";\r\n}\r\n// main application exports\r\ndeclare const __SYNC_VERSION__: string\r\nexport function getSyncVersion(): string {\r\n // injected at build time\r\n return __SYNC_VERSION__;\r\n}\r\nexport class APIResult {\r\n result: boolean;\r\n status: number;\r\n error: string;\r\n version: string;\r\n array: Array<Object> | null;\r\n constructor() { this.result = true; this.status = 200; this.error = \"\"; this.version = getSyncVersion(); this.array = null; }\r\n}\r\nexport class azureConfig {\r\n // azure graph REST API endpoints\r\n static azureElevateAccess: string = \"https://management.azure.com/providers/Microsoft.Authorization/elevateAccess?api-version=2016-07-01\";\r\n static azureListRootAssignments: string = \"https://management.azure.com/providers/Microsoft.Authorization/roleAssignments?api-version=2022-04-01&$filter=principalId+eq+\";\r\n};\r\nexport class mindlineConfig {\r\n static environmentTag: string = \"dev\";\r\n // config API endpoints\r\n static adminEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/admin`\r\n };\r\n static adminIncompleteEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/incomplete-admin`;\r\n };\r\n static adminsEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/admins`;\r\n };\r\n static auditConfigEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/audit-configuration`;\r\n }\r\n static auditEventsEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/audit-events`;\r\n };\r\n static configConsentEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/configuration/consent`;\r\n };\r\n static configEnabledEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/configuration/status`;\r\n };\r\n static configEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/configuration`;\r\n };\r\n static configsEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/configurations`;\r\n };\r\n static initEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/configuration/init`;\r\n };\r\n static readerStartSyncEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/startSync`;\r\n };\r\n static tenantEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/tenant`;\r\n };\r\n static tenantsEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/tenants`;\r\n };\r\n static workspaceEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/workspace`;\r\n };\r\n static workspacesEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/workspaces`;\r\n };\r\n static workspaceConfigsEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/workspace-configurations`;\r\n };\r\n // SignalR endpoint\r\n static signalREndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-signalrdispatcher-westus.azurewebsites.net/statsHub`;\r\n };\r\n}\r\nexport class graphConfig {\r\n // graph API predicates\r\n static graphGroupsPredicate: string = \"v1.0/groups\";\r\n static graphMailPredicate: string = \"v1.0/me/messages\";\r\n static graphMePredicate: string = \"v1.0/me\";\r\n static graphOauth2PermissionGrantsPredicate: string = \"v1.0/oauth2PermissionGrants\";\r\n static graphServicePrincipalsPredicate: string = \"v1.0/servicePrincipals\";\r\n static graphUsersPredicate: string = \"v1.0/users\";\r\n static graphTenantByDomainPredicate: string = \"beta/tenantRelationships/findTenantInformationByDomainName\";\r\n static graphTenantByIdPredicate: string = \"beta/tenantRelationships/findTenantInformationByTenantId\";\r\n // authority values are based on the well-known OIDC auth endpoints\r\n static authorityWW: string = \"https://login.microsoftonline.com/\";\r\n static authorityWWRegex: RegExp = /^(https:\\/\\/login\\.microsoftonline\\.com\\/)([\\dA-Fa-f]{8}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{12})\\/oauth2\\/authorize$/;\r\n static authorityUS: string = \"https://login.microsoftonline.us/\";\r\n static authorityUSRegex: RegExp = /^(https:\\/\\/login\\.microsoftonline\\.us\\/)([\\dA-Fa-f]{8}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{12})\\/oauth2\\/authorize$/;\r\n static authorityCN: string = \"https://login.partner.microsoftonline.cn/\";\r\n static authorityCNRegex: RegExp = /^(https:\\/\\/login\\.partner\\.microsoftonline\\.cn\\/)([\\dA-Fa-f]{8}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{12})\\/oauth2\\/authorize$/;\r\n};\r\nexport class Group {\r\n id: string = \"\";\r\n displayName: string = \"\";\r\n description: string = \"\";\r\n}\r\nexport class UserScope {\r\n group: string = \"\";\r\n value: string = \"\";\r\n consented: boolean = false;\r\n removable: boolean = false;\r\n expanded: string = \"\";\r\n static compareByValue(a: UserScope, b: UserScope): number {\r\n return a.value.localeCompare(b.value);\r\n }\r\n static compareByGroup(a: UserScope, b: UserScope): number {\r\n return a.group.localeCompare(b.group);\r\n }\r\n}\r\nexport class User {\r\n oid: string;\r\n name: string;\r\n mail: string;\r\n authority: string;\r\n tid: string;\r\n companyName: string;\r\n companyDomain: string;\r\n workspaceIDs: string;\r\n session: string; // button text\r\n spacode: string; // to get front end access token\r\n graphAccessToken: string; // front end graph access token\r\n mindlineAccessToken: string; // front end mindline access token\r\n azureAccessToken: string; // front end azure access token\r\n loginHint: string; // to help sign out without prompt\r\n scopes: string[]; // to detect if incremental consent has happened\r\n authTS: Date; // timestamp user was authenticated\r\n sel: boolean; // selection state\r\n loggingIn: boolean; // set when the user is logging in to support spinner visibility\r\n constructor() {\r\n this.oid = \"\";\r\n this.name = \"\";\r\n this.mail = \"\";\r\n this.authority = \"\";\r\n this.tid = \"\";\r\n this.companyName = \"\";\r\n this.companyDomain = \"\";\r\n this.workspaceIDs = \"\";\r\n this.session = \"Sign In\";\r\n this.spacode = \"\";\r\n this.graphAccessToken = \"\";\r\n this.mindlineAccessToken = \"\";\r\n this.azureAccessToken = \"\";\r\n this.loginHint = \"\";\r\n this.scopes = new Array();\r\n this.authTS = new Date(0);\r\n this.sel = false;\r\n this.loggingIn = false;\r\n }\r\n}\r\n\r\nexport enum TenantType {\r\n invalid = 0,\r\n aad = 1,\r\n ad = 2,\r\n googleworkspace = 3\r\n}\r\ntype TenantTypeStrings = keyof typeof TenantType;\r\nexport enum TenantPermissionType {\r\n read = 1,\r\n write = 2,\r\n notassigned = 3\r\n}\r\nexport type TenantPermissionTypeStrings = keyof typeof TenantPermissionType;\r\nexport class Tenant {\r\n tid: string;\r\n name: string;\r\n domain: string;\r\n tenantType: TenantTypeStrings;\r\n authority: string;\r\n workspaceIDs: string;\r\n sel: boolean; // selection state\r\n graphSP: string; // graph resource ID (service principal) for this tenant\r\n lookupfield: string; // are we validating by TID or by domain? \"Domain\" | \"Tenant ID\"\r\n constructor() {\r\n this.tid = \"\";\r\n this.name = \"\";\r\n this.domain = \"\";\r\n this.tenantType = \"aad\";\r\n this.authority = \"\";\r\n this.workspaceIDs = \"\";\r\n this.sel = false;\r\n this.graphSP = \"\";\r\n this.lookupfield = \"Domain\";\r\n }\r\n}\r\nfunction getAppId(authority: string, tct: TenantConfigType): string {\r\n switch (authority) {\r\n case graphConfig.authorityWW: return tct === TenantConfigType.source ? \"85d35da2-4118-4b03-aa05-605cedd7f2f8\" : \"63100afe-506e-4bb2-8ff7-d8d5ab373129\";\r\n case graphConfig.authorityUS: return tct === TenantConfigType.source ? \"b08630c7-e227-4215-9746-afc9286fb864\" : \"17aa5d5a-f09f-4cec-87a6-28596f9fa513\";\r\n case graphConfig.authorityCN: return tct === TenantConfigType.source ? \"7db7293b-add9-4a3f-8562-1a20bfe27d5e\" : \"debd015b-1154-4111-a4cb-fc220a537697\";\r\n default: debugger; return \"\";\r\n }\r\n}\r\n// https://learn.microsoft.com/en-us/graph/deployments\r\nfunction getGraphEndpoint(authority: string): string {\r\n switch (authority) {\r\n case graphConfig.authorityWW: return \"https://graph.microsoft.com/\";\r\n case graphConfig.authorityUS: return \"https://graph.microsoft.us/\";\r\n case graphConfig.authorityCN: return \"https://microsoftgraph.chinacloudapi.cn/\";\r\n default: debugger; return \"\";\r\n }\r\n}\r\n// https://learn.microsoft.com/en-us/entra/identity-platform/authentication-national-cloud\r\nfunction getLoginEndpoint(authority: string): string {\r\n switch (authority) {\r\n case graphConfig.authorityWW: return \"https://login.microsoftonline.com/\";\r\n case graphConfig.authorityUS: return \"https://login.microsoftonline.us/\";\r\n case graphConfig.authorityCN: return \"https://login.partner.microsoftonline.cn/\";\r\n default: debugger; return \"\";\r\n }\r\n}\r\nexport enum TenantConfigType {\r\n source = 1,\r\n target = 2,\r\n sourcetarget = 3\r\n}\r\nexport type TenantConfigTypeStrings = keyof typeof TenantConfigType;\r\nexport class TenantConfigInfo {\r\n tid: string;\r\n sourceGroupId: string;\r\n sourceGroupName: string;\r\n targetGroupId: string;\r\n targetGroupName: string;\r\n configurationTenantType: TenantConfigTypeStrings;\r\n deltaToken: string;\r\n usersWritten: number;\r\n configId: string;\r\n batchId: string;\r\n isReadPermissionConsented: boolean;\r\n isWritePermissionConsented: boolean;\r\n constructor() {\r\n this.tid = \"\";\r\n this.sourceGroupId = \"\";\r\n this.sourceGroupName = \"\";\r\n this.targetGroupId = \"\";\r\n this.targetGroupName = \"\";\r\n this.configurationTenantType = \"source\";\r\n this.deltaToken = \"\";\r\n this.usersWritten = 0;\r\n this.configId = \"\";\r\n this.batchId = \"\";\r\n this.isReadPermissionConsented = false;\r\n this.isWritePermissionConsented = false;\r\n }\r\n}\r\nexport class SyncConfig {\r\n id: string;\r\n workspaceId: string;\r\n name: string;\r\n description: string;\r\n tenants: TenantConfigInfo[];\r\n isEnabled: boolean;\r\n workspaceIDs: string;\r\n sel: boolean; // selection state\r\n constructor() {\r\n this.id = \"\";\r\n this.name = \"\";\r\n this.workspaceId = \"\";\r\n this.description = \"\";\r\n this.tenants = new Array();\r\n this.isEnabled = false;\r\n this.workspaceIDs = \"\";\r\n this.sel = false;\r\n }\r\n}\r\nexport class AuditConfig {\r\n id: string;\r\n workspaceId: string;\r\n name: string;\r\n description: string;\r\n isEnabled: boolean;\r\n isReadPermissionConsented: boolean;\r\n email: string;\r\n tenantId: string;\r\n sel: boolean; // selection state\r\n constructor() {\r\n this.id = \"\";\r\n this.name = \"\";\r\n this.workspaceId = \"\";\r\n this.description = \"\";\r\n this.isEnabled = false;\r\n this.isReadPermissionConsented = false;\r\n this.email = \"\";\r\n this.tenantId = \"\";\r\n this.sel = false;\r\n }\r\n}\r\n\r\nexport type ConfigType = 'sync' | 'auditlogs';\r\n\r\nexport class Workspace {\r\n id: string;\r\n name: string;\r\n ownerid: string;\r\n associatedUsers: string[];\r\n associatedTenants: string[];\r\n associatedConfigs: string[];\r\n sel: boolean; // selection state\r\n constructor() {\r\n this.id = \"\";\r\n this.name = \"\";\r\n this.ownerid = \"\";\r\n this.associatedUsers = new Array();\r\n this.associatedTenants = new Array();\r\n this.associatedConfigs = new Array();\r\n this.sel = false;\r\n }\r\n}\r\n// check for localStorage availability\r\nfunction storageAvailable() {\r\n let storage;\r\n try {\r\n storage = window.localStorage;\r\n const x = \"__storage_test__\";\r\n storage.setItem(x, x);\r\n storage.removeItem(x);\r\n return true;\r\n } catch (e) {\r\n return (\r\n e instanceof DOMException &&\r\n // everything except Firefox\r\n (e.code === 22 ||\r\n // Firefox\r\n e.code === 1014 ||\r\n // test name field too, because code might not be present\r\n // everything except Firefox\r\n e.name === \"QuotaExceededError\" ||\r\n // Firefox\r\n e.name === \"NS_ERROR_DOM_QUOTA_REACHED\") &&\r\n // acknowledge QuotaExceededError only if there's something already stored\r\n storage &&\r\n storage.length !== 0\r\n );\r\n }\r\n}\r\nexport class InitInfo {\r\n version: string = getSyncVersion();\r\n tab: number = 0;\r\n us: User[] = [];\r\n ts: Tenant[] = [];\r\n cs: SyncConfig[] = [];\r\n acs: AuditConfig[] = [];\r\n ws: Workspace[] = [];\r\n configlevelconsent_configid: string = \"\";\r\n configlevelconsent_access: TenantConfigType = TenantConfigType.sourcetarget;\r\n constructor(bClearLocalStorage: boolean = false) {\r\n this.init(bClearLocalStorage);\r\n }\r\n // get initial data from localStorage or file\r\n init(bClearLocalStorage: boolean): void {\r\n console.log(`Calling InitInfo::init(bClearLocalStorage: ${bClearLocalStorage ? \"true\" : \"false\"})`);\r\n // if we have a non-zero value stored, read it from localStorage\r\n if (storageAvailable()) {\r\n let result = localStorage.getItem(\"InitInfo\");\r\n if (result != null && typeof result === \"string\" && result !== \"\") {\r\n let initInfoString: string = result;\r\n let iiReadFromLocalStorage: InitInfo = JSON.parse(initInfoString);\r\n if (iiReadFromLocalStorage.us.length !== 0) {\r\n if (bClearLocalStorage) {\r\n localStorage.removeItem(\"InitInfo\");\r\n }\r\n else {\r\n this.#initFromObjects(iiReadFromLocalStorage);\r\n return;\r\n }\r\n }\r\n }\r\n }\r\n // if storage unavailable or we were just asked to clear, read defaults to enable usable UI\r\n this.tab = 0;\r\n this.version = getSyncVersion();\r\n this.configlevelconsent_configid = \"\";\r\n this.configlevelconsent_access = TenantConfigType.sourcetarget;\r\n var usersString = JSON.stringify(users);\r\n var tenantsString = JSON.stringify(tenants);\r\n var configsString = JSON.stringify(configs);\r\n var workspacesString = JSON.stringify(workspaces);\r\n try {\r\n this.us = deserializeArray(User, usersString);\r\n this.ts = deserializeArray(Tenant, tenantsString);\r\n this.cs = deserializeArray(SyncConfig, configsString);\r\n this.ws = deserializeArray(Workspace, workspacesString);\r\n this.tagWithWorkspaces();\r\n } catch (e) {\r\n debugger;\r\n }\r\n }\r\n save(): void {\r\n let initInfoString: string = JSON.stringify(this);\r\n localStorage.setItem(\"InitInfo\", initInfoString);\r\n }\r\n tagWithWorkspaces(): boolean {\r\n // first clear everyone's workspaceIDs\r\n this.us.map((item) => item.workspaceIDs = \"\");\r\n this.ts.map((item) => item.workspaceIDs = \"\");\r\n this.cs.map((item) => item.workspaceIDs = \"\");\r\n // for each workspace tag WorkspaceIDs of associated Users, Tenants, Configs\r\n for (let workspace of this.ws) {\r\n // find matching Users to tag with this workspace\r\n for (let userID of workspace.associatedUsers) {\r\n let user = this.us.find((currentUser) => currentUser.oid === userID);\r\n if (user !== undefined) {\r\n // we found the user\r\n user[FILTER_FIELD] += workspace.id;\r\n user[FILTER_FIELD] += \" \";\r\n } else {\r\n // we should not have InitInfo missing Workspace components\r\n debugger;\r\n }\r\n }\r\n // find matching Tenants to tag with this workspace\r\n for (let tenantID of workspace.associatedTenants) {\r\n let tenant = this.ts.find((currentTenant) => currentTenant.tid === tenantID);\r\n if (tenant !== undefined) {\r\n // we found the tenant\r\n tenant[FILTER_FIELD] += workspace.id;\r\n tenant[FILTER_FIELD] += \" \";\r\n } else {\r\n // we should not have InitInfo missing Workspace components\r\n debugger;\r\n }\r\n }\r\n // find matching Configs to tag with this workspace\r\n for (let configID of workspace.associatedConfigs) {\r\n let config = this.cs.find((currentConfig) => currentConfig.id === configID);\r\n if (config !== undefined) {\r\n // we found the config\r\n config[FILTER_FIELD] += workspace.id;\r\n config[FILTER_FIELD] += \" \";\r\n } else {\r\n // we should not have InitInfo missing Workspace components\r\n debugger;\r\n }\r\n }\r\n }\r\n return true;\r\n }\r\n #initFromObjects(ii: InitInfo): void {\r\n this.tab = ii.tab;\r\n this.version = getSyncVersion();\r\n this.configlevelconsent_configid = ii.configlevelconsent_configid;\r\n this.configlevelconsent_access = ii.configlevelconsent_access;\r\n if (typeof ii.us === \"undefined\") {\r\n this.us = new Array<User>();\r\n }\r\n else {\r\n this.us = ii.us.map((user: User) => {\r\n let newuser: User = new User();\r\n newuser.oid = user.oid;\r\n newuser.name = user.name;\r\n newuser.mail = user.mail;\r\n newuser.authority = user.authority;\r\n newuser.tid = user.tid;\r\n newuser.companyName = user.companyName;\r\n newuser.companyDomain = user.companyDomain;\r\n newuser.workspaceIDs = user.workspaceIDs;\r\n newuser.session = user.session;\r\n newuser.spacode = user.spacode;\r\n newuser.graphAccessToken = user.graphAccessToken;\r\n newuser.mindlineAccessToken = user.mindlineAccessToken;\r\n newuser.azureAccessToken = user.azureAccessToken;\r\n newuser.loginHint = user.loginHint;\r\n newuser.scopes = user.scopes;\r\n newuser.authTS = new Date(user.authTS);\r\n newuser.sel = user.sel;\r\n newuser.loggingIn = user.loggingIn;\r\n return newuser\r\n });\r\n }\r\n if (typeof ii.ts === \"undefined\") {\r\n this.ts = new Array<Tenant>();\r\n }\r\n else {\r\n this.ts = ii.ts.map((tenant: Tenant) => {\r\n let newtenant: Tenant = new Tenant();\r\n newtenant.tid = tenant.tid;\r\n newtenant.name = tenant.name;\r\n newtenant.domain = tenant.domain;\r\n newtenant.tenantType = tenant.tenantType;\r\n newtenant.authority = tenant.authority;\r\n newtenant.workspaceIDs = tenant.workspaceIDs;\r\n newtenant.sel = tenant.sel;\r\n newtenant.lookupfield = tenant.lookupfield;\r\n return newtenant;\r\n });\r\n }\r\n if (typeof ii.cs === \"undefined\") {\r\n this.cs = new Array<SyncConfig>();\r\n }\r\n else {\r\n this.cs = ii.cs.map((config: SyncConfig) => {\r\n let newconfig: SyncConfig = new SyncConfig();\r\n newconfig.id = config.id;\r\n newconfig.workspaceId = config.workspaceId;\r\n newconfig.name = config.name;\r\n newconfig.description = config.description;\r\n newconfig.tenants = config.tenants;\r\n newconfig.isEnabled = config.isEnabled;\r\n newconfig.workspaceIDs = config.workspaceIDs;\r\n newconfig.sel = config.sel;\r\n return newconfig;\r\n });\r\n }\r\n if (typeof ii.ws === \"undefined\") {\r\n this.ws = new Array<Workspace>();\r\n }\r\n else {\r\n this.ws = ii.ws.map((workspace: Workspace) => {\r\n let newworkspace: Workspace = new Workspace();\r\n newworkspace.id = workspace.id;\r\n newworkspace.name = workspace.name;\r\n newworkspace.ownerid = workspace.ownerid;\r\n newworkspace.associatedUsers = workspace.associatedUsers;\r\n newworkspace.associatedTenants = workspace.associatedTenants;\r\n newworkspace.associatedConfigs = workspace.associatedConfigs;\r\n newworkspace.sel = workspace.sel;\r\n return newworkspace;\r\n });\r\n }\r\n }\r\n}\r\nconst tasksData: any[] = [\r\n {\r\n id: 1,\r\n task: \"initialization\",\r\n start: \"1970-01-01T00:00:00\",\r\n end: \"1970-01-01T00:00:00\",\r\n expected: \"0:22\",\r\n status: \"not started\",\r\n expanded: true,\r\n subtasks: [\r\n {\r\n id: 2,\r\n task: \"authenticate user\",\r\n start: \"1970-01-01T00:00:00\",\r\n end: \"1970-01-01T00:00:00\",\r\n expected: \"0:01\",\r\n status: \"not started\"\r\n },\r\n {\r\n id: 3,\r\n task: \"reload React\",\r\n start: \"1970-01-01T00:00:00\",\r\n end: \"1970-01-01T00:00:00\",\r\n expected: \"0:07\",\r\n status: \"not started\"\r\n },\r\n {\r\n id: 4,\r\n task: \"GET tenant details\",\r\n start: \"1970-01-01T00:00:00\",\r\n end: \"1970-01-01T00:00:00\",\r\n expected: \"0:01\",\r\n status: \"not started\"\r\n },\r\n {\r\n id: 5,\r\n task: \"POST config init\",\r\n start: \"1970-01-01T00:00:00\",\r\n end: \"1970-01-01T00:00:00\",\r\n expected: \"0:10\",\r\n status: \"not started\"\r\n },\r\n {\r\n id: 6,\r\n task: \"GET workspaces\",\r\n start: \"1970-01-01T00:00:00\",\r\n end: \"1970-01-01T00:00:00\",\r\n expected: \"0:04\",\r\n status: \"not started\"\r\n }\r\n ]\r\n }\r\n];\r\nexport type TaskType = \"initialization\" |\r\n \"authenticate user\" |\r\n \"reload React\" |\r\n \"GET tenant details\" |\r\n \"POST config init\" |\r\n \"GET workspaces\";\r\nexport class TaskArray {\r\n tasks: Task[];\r\n constructor(bClearLocalStorage: boolean) {\r\n this.tasks = [new Task()];\r\n this.init(bClearLocalStorage);\r\n }\r\n // get initial data from localStorage or file\r\n init(bClearLocalStorage: boolean): void {\r\n console.log(`Calling TaskArray::init(bClearLocalStorage: ${bClearLocalStorage ? \"true\" : \"false\"})`);\r\n // first clear task array\r\n this.tasks.length = 0;\r\n // then clear localStorage if we have been asked to\r\n if (bClearLocalStorage) {\r\n if (storageAvailable()) localStorage.removeItem(\"Tasks\");\r\n }\r\n // then try localStorage\r\n if (storageAvailable()) {\r\n let result = localStorage.getItem(\"Tasks\");\r\n if (result != null && typeof result === \"string\" && result !== \"\") {\r\n // properly create Tasks and Dates from retrieved string\r\n let tasksString: string = result;\r\n let taskArray: TaskArray = JSON.parse(tasksString);\r\n this.tasks = this.#initTasksFromObjects(taskArray.tasks);\r\n let l = this.tasks.length;\r\n if (l !== 0) return;\r\n }\r\n }\r\n // if here, there was nothing in localStorage, use initialization file\r\n this.tasks = this.#initTasksFromObjects(tasksData);\r\n }\r\n // set start time for a task\r\n setTaskStart(taskType: TaskType, startDate: Date): void {\r\n let task: Task | undefined = this.#findTask(taskType);\r\n if (task != undefined && task != null) {\r\n task.setStart(startDate);\r\n task.status = \"in progress\";\r\n this.#save();\r\n }\r\n else {\r\n debugger;\r\n }\r\n }\r\n // set end time for a task\r\n setTaskEnd(taskType: TaskType, endDate: Date, status: string): void {\r\n let task: Task | undefined = this.#findTask(taskType);\r\n if (task != undefined && task != null) {\r\n task.setEnd(endDate);\r\n task.status = status;\r\n this.#save();\r\n }\r\n else {\r\n debugger;\r\n }\r\n }\r\n //\r\n // private\r\n //\r\n #findTask(taskType: TaskType): Task | undefined {\r\n let task: Task | undefined = this.tasks.find(t => t.task == taskType);\r\n if (task == undefined || task == null) {\r\n for (task of this.tasks) {\r\n if (task.subtasks != undefined && task.subtasks != null) {\r\n task = task.subtasks.find(t => t.task == taskType);\r\n if (task != undefined && task != null) break;\r\n }\r\n }\r\n }\r\n return task;\r\n }\r\n #initTasksFromObjects(tasks: Task[]): Task[] {\r\n return tasks.map((t: Task) => {\r\n let newTask: Task = new Task();\r\n newTask.id = t.id;\r\n newTask.task = t.task;\r\n newTask.setStart(new Date(t.start));\r\n newTask.setEnd(new Date(t.end));\r\n newTask.expected = t.expected;\r\n newTask.status = t.status;\r\n newTask.expanded = t.expanded;\r\n if (typeof t.subtasks !== \"undefined\" && t.subtasks != null) {\r\n newTask.subtasks = t.subtasks.map((st: Task) => {\r\n let newSubtask: Task = new Task();\r\n newSubtask.id = st.id;\r\n newSubtask.task = st.task;\r\n newSubtask.setStart(new Date(st.start))\r\n newSubtask.setEnd(new Date(st.end));\r\n newSubtask.expected = st.expected;\r\n newSubtask.status = st.status;\r\n newSubtask.expanded = st.expanded;\r\n return newSubtask;\r\n })\r\n }\r\n return newTask;\r\n });\r\n }\r\n #save(): void {\r\n let taskArrayString: string = JSON.stringify(this);\r\n if (storageAvailable()) {\r\n localStorage.setItem(\"Tasks\", taskArrayString);\r\n }\r\n }\r\n}\r\nexport class Task {\r\n id: number = 0;\r\n task: string = \"\";\r\n start: Date = new Date();\r\n startDisplay: string = \"\";\r\n end: Date = new Date();\r\n endDisplay: string = \"\";\r\n elapsedDisplay: string = \"\";\r\n expected: number = 0;\r\n status: string = \"\";\r\n expanded: boolean = false;\r\n subtasks: Task[] = [];\r\n setEnd(endDate: Date): void {\r\n this.end = endDate;\r\n this.endDisplay = `${this.end.getMinutes().toString().padStart(2, \"0\")}:${this.end.getSeconds().toString().padStart(2, \"0\")}`;\r\n let minuteAdjustment: number = 0;\r\n let elapsedSeconds: number = this.end.getSeconds() - this.start.getSeconds();\r\n if (elapsedSeconds < 0) { elapsedSeconds += 60; minuteAdjustment = -1; }\r\n let elapsedMinutes: number = this.end.getMinutes() - this.start.getMinutes() + minuteAdjustment;\r\n if (elapsedMinutes < 0) elapsedMinutes += 60;\r\n this.elapsedDisplay = `${elapsedMinutes.toString().padStart(2, \"0\")}:${elapsedSeconds.toString().padStart(2, \"0\")}`;\r\n };\r\n setStart(startDate: Date): void {\r\n this.start = startDate;\r\n this.startDisplay = `${this.start.getMinutes().toString().padStart(2, \"0\")}:${this.start.getSeconds().toString().padStart(2, \"0\")}`;\r\n };\r\n}\r\n// class corresponding to an execution of a SyncConfig - a *TenantNode* for each source tenant, each with a *TenantNode* array of target tenants\r\nexport class Milestone {\r\n Run: number = 0;\r\n Start: Date = new Date();\r\n startDisplay: string = \"\";\r\n POST: Date;\r\n postDisplay: string;\r\n Read: Date;\r\n readDisplay: string;\r\n Write: Date;\r\n writeDisplay: string;\r\n Duration: Date;\r\n durationDisplay: string;\r\n constructor(run: number) {\r\n this.Run = run;\r\n this.start(\"\");\r\n this.POST = new Date();\r\n this.postDisplay = \"\";\r\n this.Read = new Date();\r\n this.readDisplay = \"\";\r\n this.Write = new Date();\r\n this.writeDisplay = \"\";\r\n this.Duration = new Date();\r\n this.durationDisplay = \"\";\r\n }\r\n start(start: string): void {\r\n start == \"\" ? this.Start = new Date() : this.Start = new Date(start);\r\n this.startDisplay = `${this.Start.getMinutes().toString().padStart(2, \"0\")}:${this.Start.getSeconds().toString().padStart(2, \"0\")}`;\r\n }\r\n post(post: string): void {\r\n post == \"\" ? this.POST = new Date() : this.POST = new Date(post);\r\n this.postDisplay = `${this.POST.getMinutes().toString().padStart(2, \"0\")}:${this.POST.getSeconds().toString().padStart(2, \"0\")}`;\r\n }\r\n read(read: string): void {\r\n read == \"\" ? this.Read = new Date() : this.Read = new Date(read);\r\n this.readDisplay = `${this.Read.getMinutes().toString().padStart(2, \"0\")}:${this.Read.getSeconds().toString().padStart(2, \"0\")}`;\r\n }\r\n write(write: string): void {\r\n write == \"\" ? this.Write = new Date() : this.Write = new Date(write);\r\n this.writeDisplay = `${this.Write.getMinutes().toString().padStart(2, \"0\")}:${this.Write.getSeconds().toString().padStart(2, \"0\")}`;\r\n this.Duration = new Date(this.Write.getTime() - this.Start.getTime());\r\n this.durationDisplay = `${this.Duration.getMinutes().toString().padStart(2, \"0\")}:${this.Duration.getSeconds().toString().padStart(2, \"0\")}`;\r\n }\r\n}\r\nexport class MilestoneArray {\r\n milestones: Milestone[] = [];\r\n constructor(bClearLocalStorage: boolean) {\r\n this.init(bClearLocalStorage);\r\n }\r\n init(bClearLocalStorage: boolean): void {\r\n // read from localstorage by default\r\n if (storageAvailable()) {\r\n let result = localStorage.getItem(\"syncmilestones\");\r\n if (result != null && typeof result === \"string\" && result !== \"\") {\r\n let milestonesString: string = result;\r\n let milestones: Object [] = JSON.parse(milestonesString);\r\n if (milestones.length !== 0) {\r\n if (bClearLocalStorage) {\r\n localStorage.removeItem(\"syncmilestones\");\r\n }\r\n else {\r\n this.#initFromObjects(milestones);\r\n return;\r\n }\r\n }\r\n }\r\n }\r\n // if storage unavailable or we were just asked to clear, read from default syncmilestone file\r\n this.#initFromObjects(syncmilestones);\r\n }\r\n save(): void {\r\n let milestonesString: string = JSON.stringify(this.milestones);\r\n if (storageAvailable()) {\r\n localStorage.setItem(\"syncmilestones\", milestonesString);\r\n }\r\n }\r\n // milestone tracking during a sync\r\n start(setMilestones: (milestones: Milestone[]) => void): void {\r\n // we should always have a milestone array and a first milestone\r\n if (this.milestones == null || this.milestones.length < 1) { debugger; return; }\r\n let currentRun: number = Number(this.milestones[0].Run);\r\n // create a new milestone and prepend to front of array\r\n let newMilestone: Milestone = new Milestone(currentRun+1);\r\n this.milestones.unshift(newMilestone);\r\n // re-define milestone array to trigger render\r\n this.milestones = this.milestones.map((ms: Milestone) => {\r\n let newms = new Milestone(ms.Run);\r\n newms.Start = ms.Start;\r\n newms.startDisplay = ms.startDisplay;\r\n newms.POST = ms.POST;\r\n newms.postDisplay = ms.postDisplay;\r\n newms.Read = ms.Read;\r\n newms.readDisplay = ms.readDisplay;\r\n newms.Write = ms.Write;\r\n newms.writeDisplay = ms.writeDisplay;\r\n newms.Duration = ms.Duration;\r\n newms.durationDisplay = ms.durationDisplay;\r\n return newms;\r\n });\r\n setMilestones(this.milestones);\r\n console.log(`Start milestone: ${this.milestones[0].Run}:${this.milestones[0].Start}`);\r\n }\r\n unstart(setMilestones: (milestones: Milestone[]) => void): void {\r\n // we should always have a milestone array and a first milestone\r\n if (this.milestones == null || this.milestones.length < 1) { debugger; return; }\r\n // remove first milestone from front of array\r\n let removedMilestone: Milestone | undefined = this.milestones.shift();\r\n // re-define milestone array to trigger render\r\n this.milestones = this.milestones.map((ms: Milestone) => {\r\n let newms = new Milestone(ms.Run);\r\n newms.Start = ms.Start;\r\n newms.startDisplay = ms.startDisplay;\r\n newms.POST = ms.POST;\r\n newms.postDisplay = ms.postDisplay;\r\n newms.Read = ms.Read;\r\n newms.readDisplay = ms.readDisplay;\r\n newms.Write = ms.Write;\r\n newms.writeDisplay = ms.writeDisplay;\r\n newms.Duration = ms.Duration;\r\n newms.durationDisplay = ms.durationDisplay;\r\n return newms;\r\n });\r\n setMilestones(this.milestones);\r\n console.log(`Unstart removed first milestone: ${removedMilestone!.Run}:${removedMilestone!.Start}`);\r\n }\r\n post(setMilestones: (milestones: Milestone[]) => void): void {\r\n // update the post value of the first milestone\r\n if (this.milestones == null || this.milestones.length < 1) { debugger; return; }\r\n this.milestones[0].post(\"\");\r\n setMilestones(this.milestones);\r\n console.log(`POST milestone: ${this.milestones[0].Run}:${this.milestones[0].POST}`);\r\n }\r\n read(setMilestones: (milestones: Milestone[]) => void): void {\r\n if (this.milestones == null || this.milestones.length < 1) { debugger; return; }\r\n this.milestones[0].read(\"\");\r\n setMilestones(this.milestones);\r\n console.log(`Read milestone: ${this.milestones[0].Run}:${this.milestones[0].Read}`);\r\n }\r\n write(setMilestones: (milestones: Milestone[]) => void): void {\r\n if (this.milestones == null || this.milestones.length < 1) { debugger; return; }\r\n this.milestones[0].write(\"\");\r\n // while we have >10 complete milestones, remove the last\r\n while (this.milestones.length > 10) {\r\n let removed: Milestone | undefined = this.milestones.pop();\r\n console.log(`Removed milestone #${removed!.Run}: ${removed!.Start}`);\r\n }\r\n // save to localstorage\r\n this.save();\r\n // re-define milestone array to trigger render\r\n this.milestones = this.milestones.map((ms: Milestone) => {\r\n let newms = new Milestone(ms.Run);\r\n newms.Start = ms.Start;\r\n newms.startDisplay = ms.startDisplay;\r\n newms.POST = ms.POST;\r\n newms.postDisplay = ms.postDisplay;\r\n newms.Read = ms.Read;\r\n newms.readDisplay = ms.readDisplay;\r\n newms.Write = ms.Write;\r\n newms.writeDisplay = ms.writeDisplay;\r\n newms.Duration = ms.Duration;\r\n newms.durationDisplay = ms.durationDisplay;\r\n return newms;\r\n });\r\n setMilestones(this.milestones);\r\n }\r\n #initFromObjects(milestones: Object[]): void {\r\n if (milestones == null) {\r\n this.milestones = new Array();\r\n }\r\n else {\r\n this.milestones = milestones.map((milestone: any) => {\r\n let ms: Milestone = new Milestone(Number(milestone.Run));\r\n ms.start(milestone.Start);\r\n ms.post(milestone.POST);\r\n ms.read(milestone.Read);\r\n ms.write(milestone.Write);\r\n return ms;\r\n });\r\n }\r\n }\r\n}\r\nexport class BatchArray {\r\n tenantNodes: TenantNode[];\r\n pb_startTS: number;\r\n pb_progress: number;\r\n pb_increment: number;\r\n pb_idle: number;\r\n pb_idleMax: number;\r\n pb_total: number;\r\n pb_timer: NodeJS.Timeout | null;\r\n milestoneArray: MilestoneArray;\r\n constructor(\r\n config: SyncConfig | null,\r\n syncPortalGlobalState: InitInfo | null,\r\n bClearLocalStorage: boolean\r\n ) {\r\n this.tenantNodes = new Array<TenantNode>();\r\n this.init(config, syncPortalGlobalState, bClearLocalStorage);\r\n this.pb_startTS = 0;\r\n this.pb_progress = 0;\r\n this.pb_increment = 0;\r\n this.pb_timer = null;\r\n this.pb_idle = 0;\r\n this.pb_idleMax = 0;\r\n this.pb_total = 0;\r\n this.milestoneArray = new MilestoneArray(false);\r\n }\r\n // populate tenantNodes based on config tenants\r\n init(\r\n config: SyncConfig | null | undefined,\r\n syncPortalGlobalState: InitInfo | null,\r\n bClearLocalStorage: boolean\r\n ): void {\r\n console.log(\r\n `Calling BatchArray::init(config: \"${config ? config.name : \"null\"}\", bClearLocalStorage: ${bClearLocalStorage ? \"true\" : \"false\"})`\r\n );\r\n // clear localStorage if we have been asked to\r\n if (bClearLocalStorage) {\r\n if (storageAvailable()) {\r\n localStorage.removeItem(config!.name);\r\n this.milestoneArray.init(bClearLocalStorage);\r\n }\r\n }\r\n // create BatchArray if passed SyncConfig and InitInfo\r\n if (config != null &&\r\n config.tenants != null &&\r\n syncPortalGlobalState != null) {\r\n // clear batch array only if we have been passed something with which to replace it\r\n this.tenantNodes.length = 0;\r\n // create a sourceTenantNode for each Source and SourceTarget\r\n config.tenants.map((tciPotentialSource: TenantConfigInfo) => {\r\n if (\r\n tciPotentialSource.configurationTenantType === \"source\" ||\r\n tciPotentialSource.configurationTenantType === \"sourcetarget\"\r\n ) {\r\n let sourceTenant = syncPortalGlobalState.ts.find(\r\n (t) => t.tid === tciPotentialSource.tid\r\n );\r\n if (sourceTenant != null) {\r\n let sourceTenantNode: TenantNode = new TenantNode(\r\n tciPotentialSource.tid,\r\n sourceTenant.name,\r\n tciPotentialSource.batchId\r\n );\r\n this.tenantNodes.push(sourceTenantNode);\r\n } else {\r\n console.log(\r\n `Error: no tenant found for config source tenant ${config.name}`\r\n );\r\n debugger;\r\n return;\r\n }\r\n }\r\n });\r\n // create targetTenantNodes for each non-matching Target and SourceTarget\r\n this.tenantNodes.map((sourceTenantNode: TenantNode) => {\r\n config.tenants.map((tciPotentialTarget: TenantConfigInfo) => {\r\n // is this a valid target?\r\n if (\r\n tciPotentialTarget.configurationTenantType === \"target\" ||\r\n tciPotentialTarget.configurationTenantType === \"sourcetarget\"\r\n ) {\r\n // is this a valid target that does not match this source?\r\n if (tciPotentialTarget.tid !== sourceTenantNode.tid) {\r\n let targetTenant = syncPortalGlobalState.ts.find(\r\n (t) => t.tid === tciPotentialTarget.tid\r\n );\r\n if (targetTenant != null) {\r\n let targetTenantNode: TenantNode = new TenantNode(\r\n tciPotentialTarget.tid,\r\n targetTenant.name,\r\n tciPotentialTarget.batchId\r\n );\r\n sourceTenantNode.targets.push(targetTenantNode);\r\n sourceTenantNode.expanded = true;\r\n } else {\r\n console.log(\r\n `Error: no tenant found for config target tenant ${config.name}`\r\n );\r\n debugger;\r\n return;\r\n }\r\n }\r\n }\r\n });\r\n });\r\n }\r\n }\r\n initializeProgressBar(setSyncProgress: (progress: number) => void, setConfigSyncResult: (result: string) => void, setIdleText: (idleText: string) => void, setMilestones: (milestones: Milestone[]) => void): void {\r\n this.pb_startTS = Date.now();\r\n this.pb_progress = 0;\r\n this.pb_increment = .25;\r\n this.pb_idle = 0;\r\n this.pb_idleMax = 0;\r\n this.pb_total = 0;\r\n this.pb_timer = setInterval(() => {\r\n // if signalR has finished the sync, stop the timer\r\n if (this.milestoneArray.milestones[0].Write != null) {\r\n clearInterval(this.pb_timer!);\r\n this.pb_timer = null;\r\n this.pb_progress = 100;\r\n setSyncProgress(this.pb_progress);\r\n setIdleText(`Complete. [max idle: ${this.pb_idleMax}]`);\r\n }\r\n else {\r\n // if we've gone 60 seconds without a signalR message, finish the sync\r\n this.pb_total = this.pb_total + 1;\r\n this.pb_idle = this.pb_idle + 1;\r\n this.pb_idleMax = Math.max(this.pb_idle, this.pb_idleMax);\r\n setIdleText(`${this.pb_total} seconds elapsed. Last update ${this.pb_idle} seconds ago. [max idle: ${this.pb_idleMax}/60]`);\r\n if (this.pb_idle >= 60) {\r\n if (this.milestoneArray.milestones[0].Write == null) {\r\n //this.milestoneArray.write(setMilestones); -- allow sync to cntinue\r\n setConfigSyncResult(`sync continuing, but no update for ${this.pb_idle} seconds`);\r\n }\r\n }\r\n // if we get to 100, the progress bar stops but SignalR or countdown timer completes the sync\r\n if (this.pb_progress < 100) {\r\n this.pb_progress = Math.min(100, this.pb_progress + this.pb_increment);\r\n setSyncProgress(this.pb_progress);\r\n }\r\n }\r\n }, 1000);\r\n this.milestoneArray.start(setMilestones);\r\n }\r\n uninitializeProgressBar(setSyncProgress: (progress: number) => void, setConfigSyncResult: (result: string) => void, setIdleText: (idleText: string) => void, setMilestones: (milestones: Milestone[]) => void): void {\r\n this.pb_startTS = 0;\r\n this.pb_progress = 0;\r\n setSyncProgress(this.pb_progress);\r\n setConfigSyncResult(\"sync failed to execute\");\r\n this.pb_increment = 0;\r\n clearInterval(this.pb_timer!);\r\n this.pb_timer = null;\r\n this.pb_idle = 0;\r\n this.pb_idleMax = 0;\r\n setIdleText(`No updates seen for ${this.pb_idle} seconds. [max idle: ${this.pb_idleMax}]`);\r\n this.milestoneArray.unstart(setMilestones);\r\n }\r\n initializeSignalR(\r\n config: SyncConfig | null | undefined,\r\n syncPortalGlobalState: InitInfo | null,\r\n batchIdArray: Array<Object>,\r\n setRefreshDeltaTrigger: (workspace: string) => void,\r\n setReadersTotal: (readersTotal: number) => void,\r\n setReadersCurrent: (readersCurrent: number) => void,\r\n setWritersTotal: (writersTotal: number) => void,\r\n setWritersCurrent: (writersCurrent: number) => void ,\r\n setMilestones: (milestones: Milestone[]) => void,\r\n setConfigSyncResult: (result: string) => void,\r\n bClearLocalStorage: boolean\r\n ): void {\r\n bClearLocalStorage = bClearLocalStorage;\r\n // we have just completed a successful POST to startSync\r\n this.milestoneArray.post(setMilestones);\r\n setConfigSyncResult(\"started sync, waiting for updates...\");\r\n // re-initialize batch array with Configuration updated by the succcessful POST to startSync\r\n this.init(config, syncPortalGlobalState, false);\r\n // define newMessage handler that can access *this*\r\n let handler = (message: string) => {\r\n console.log(message);\r\n let item = JSON.parse(message);\r\n // reset the countdown timer every time we get a message\r\n this.pb_idle = 0;\r\n // find the associated tenant for this SignalR message\r\n let matchingPair: any | undefined = batchIdArray.find((o: any) => o.BatchId == item.TargetID);\r\n if (matchingPair == null) {\r\n console.log(`Batch ${item.TargetID} not found in batchIdArray.`);\r\n debugger;\r\n return;\r\n }\r\n let tenantNode: TenantNode | undefined = this.tenantNodes.find((t: TenantNode) => t.tid === matchingPair.SourceId);\r\n if (tenantNode == null) { // null OR undefined\r\n console.log(`Tenant ${matchingPair.SourceId} not found in BatchArray.`);\r\n debugger;\r\n return;\r\n }\r\n tenantNode.batchId = matchingPair.BatchId;\r\n // process stats for this SignalR message (one batch per tenant node)\r\n let statsarray = item.Stats; // get the array of statistics\r\n let statskeys = Object.keys(statsarray); // get the keys of the array\r\n let statsvalues = Object.values(statsarray); // get the values of the array\r\n // does this tenantnode/batch have nothing to sync?\r\n let bTotalCountZero: boolean = false;\r\n let bCurrentCountZero: boolean = false;\r\n for (let j = 0; j < statskeys.length; j++) {\r\n let bTotalCount = statskeys[j].endsWith(\"TotalCount\");\r\n let bCurrentCount = statskeys[j].endsWith(\"CurrentCount\");\r\n let bDeferredCount = statskeys[j].endsWith(\"DeferredCount\");\r\n let bRescheduledCount = statskeys[j].endsWith(\"RescheduledCount\");\r\n if (statskeys[j].startsWith(\"Reader\")) {\r\n // parse tid from Reader key\r\n let tidRegexp = /Reader\\/TID:(.+)\\/TotalCount/;\r\n if (bCurrentCount) tidRegexp = /Reader\\/TID:(.+)\\/CurrentCount/;\r\n if (bDeferredCount) tidRegexp = /Reader\\/TID:(.+)\\/DeferredCount/;\r\n if (bRescheduledCount) tidRegexp = /Reader\\/TID:(.+)\\/RescheduledCount/;\r\n let matchTID = statskeys[j].match(tidRegexp);\r\n if (matchTID == null) {\r\n console.log(`tid not found in ${statskeys[j]}.`);\r\n debugger;\r\n return;\r\n }\r\n if (bTotalCount) {\r\n bTotalCountZero = Number(statsvalues[j]) == 0;\r\n tenantNode.total = Math.max(Number(statsvalues[j]), tenantNode.total);\r\n console.log(`----- ${tenantNode.name} TID: ${tenantNode.tid} batchId: ${tenantNode.batchId}`);\r\n console.log(`----- ${tenantNode.name} Total To Read: ${tenantNode.total}`);\r\n }\r\n else {\r\n bCurrentCountZero = Number(statsvalues[j]) == 0;\r\n tenantNode.read = Math.max(Number(statsvalues[j]), tenantNode.read);\r\n console.log(`----- ${tenantNode.name} Currently Read: ${tenantNode.read}`);\r\n }\r\n }\r\n tenantNode.nothingtosync = bTotalCountZero && bCurrentCountZero;\r\n if (statskeys[j].startsWith(\"Writer\")) {\r\n // parse tid from Writer key\r\n let tidRegexp = /Writer\\/TID:(.+)\\/TotalCount/;\r\n if (bCurrentCount) tidRegexp = /Writer\\/TID:(.+)\\/CurrentCount/;\r\n if (bDeferredCount) tidRegexp = /Writer\\/TID:(.+)\\/DeferredCount/;\r\n if (bRescheduledCount) tidRegexp = /Writer\\/TID:(.+)\\/RescheduledCount/;\r\n let matchTID: RegExpMatchArray | null = statskeys[j].match(tidRegexp);\r\n if (matchTID == null) {\r\n console.log(`tid not found in ${statskeys[j]}.`);\r\n debugger;\r\n return;\r\n }\r\n // this Writer node should exist precisely under the Reader for this SignalR message\r\n let writerNode: TenantNode | undefined = tenantNode.targets.find((t: TenantNode) => t.tid === matchTID[1]);\r\n if (writerNode == null) {\r\n console.log(`Writer ${tenantNode.name} not found under Reader ${tenantNode.name}.`);\r\n debugger;\r\n return;\r\n }\r\n writerNode.batchId = matchingPair.BatchId;\r\n if (bTotalCount) {\r\n writerNode.total = Math.max(Number(statsvalues[j]), writerNode.total);\r\n console.log(`----- ${writerNode.name} TID: ${writerNode.tid} batchId: ${writerNode.batchId}`);\r\n console.log(`----- ${writerNode.name} Total To Write: ${writerNode.total}`);\r\n }\r\n else if (bCurrentCount) {\r\n writerNode.written = Math.max(Number(statsvalues[j]), writerNode.written);\r\n console.log(`----- ${writerNode.name} Total Written: ${writerNode.written}`);\r\n }\r\n else if (bDeferredCount || bRescheduledCount) {\r\n writerNode.deferred = Math.max(Number(statsvalues[j]), writerNode.deferred);\r\n console.log(`----- ${writerNode.name} Total Deferred: ${writerNode.deferred}`);\r\n }\r\n else {\r\n console.log(`unknown writer type`);\r\n debugger;\r\n return;\r\n }\r\n writerNode.update(writerNode.total, writerNode.read, writerNode.written, writerNode.deferred);\r\n }\r\n }\r\n // update status based on all updates in this message\r\n tenantNode.update(tenantNode.total, tenantNode.read, tenantNode.written, tenantNode.deferred);\r\n // for each message, enumerate nodes to assess completion state\r\n let bReadingComplete: boolean = true;\r\n let bWritingComplete: boolean = true;\r\n let bWritingStarted: boolean = false;\r\n let bNothingToSync: boolean = true;\r\n let readerTotal: number = 0;\r\n let readerCurrent: number = 0;\r\n let writerTotal: number = 0;\r\n let writerCurrent: number = 0;\r\n this.tenantNodes.map((sourceTenantNode: TenantNode) => {\r\n sourceTenantNode.targets.map((writerNode: TenantNode) => {\r\n bWritingComplete &&= (writerNode.status == \"complete\" || writerNode.status == \"failed\");\r\n bWritingStarted ||= (writerNode.total > 0 || writerNode.status != \"not started\");\r\n writerTotal += Math.max(writerNode.total, sourceTenantNode.total);\r\n writerCurrent += writerNode.written;\r\n });\r\n bNothingToSync &&= sourceTenantNode.nothingtosync;\r\n bReadingComplete &&= (sourceTenantNode.status == \"complete\" || sourceTenantNode.status == \"failed\");\r\n readerTotal += sourceTenantNode.total;\r\n readerCurrent += sourceTenantNode.read;\r\n });\r\n // set linear gauge max and current values\r\n setReadersTotal(readerTotal);\r\n setReadersCurrent(readerCurrent);\r\n setWritersTotal(Math.max(writerTotal, readerTotal));\r\n setWritersCurrent(writerCurrent);\r\n // check to see if there was nothing to sync\r\n if (bNothingToSync) {\r\n this.milestoneArray.write(setMilestones);\r\n setConfigSyncResult(\"nothing to sync\");\r\n console.log(`Setting config sync result: \"nothing to sync\"`);\r\n }\r\n else {\r\n // because it is an important milestone, we always check if we have *just* completed reading\r\n if (bReadingComplete && this.milestoneArray.milestones[0].Read == null) {\r\n this.milestoneArray.read(setMilestones);\r\n setConfigSyncResult(\"reading complete\");\r\n console.log(`Setting config sync result: \"reading complete\"`);\r\n // trigger refresh delta tokens\r\n setRefreshDeltaTrigger(config!.workspaceId);\r\n // change to % per second to complete in 12x as long as it took to get here\r\n let readTS = Date.now();\r\n let secsElapsed = (readTS - this.pb_startTS) / 1000;\r\n let expectedPercentDone = 8.5;\r\n let expectedPercentPerSecond = secsElapsed / expectedPercentDone;\r\n this.pb_increment = expectedPercentPerSecond;\r\n console.log(`Setting increment: ${this.pb_increment}% per second`);\r\n }\r\n // with that out of the way, is writing complete?\r\n if (bWritingComplete) {\r\n this.milestoneArray.write(setMilestones);\r\n setConfigSyncResult(\"sync complete\");\r\n console.log(`Setting config sync result: \"complete\"`);\r\n }\r\n // if not, has writing even started?\r\n else if (bWritingStarted) {\r\n setConfigSyncResult(\"writing in progress\");\r\n console.log(`Setting config sync result: \"writing in progress\"`);\r\n }\r\n // else, we must be reading (unless we already completed reading)\r\n else if (this.milestoneArray.milestones[0].Read == null) {\r\n setConfigSyncResult(\"reading in progress\");\r\n console.log(`Setting config sync result: \"reading in progress\"`);\r\n }\r\n }\r\n }\r\n // start SignalR connection based on each batchId\r\n batchIdArray.map((batchPair: any) => {\r\n const endpoint: string = mindlineConfig.signalREndpoint();\r\n let endpointUrl: URL = new URL(endpoint);\r\n endpointUrl.searchParams.append(\"statsId\", batchPair.BatchId);\r\n console.log(`Creating SignalR Hub for TID: ${batchPair.SourceId} ${endpointUrl.href}`);\r\n const connection: signalR.HubConnection = new signalR.HubConnectionBuilder()\r\n .withUrl(endpointUrl.href)\r\n .withAutomaticReconnect()\r\n .configureLogging(signalR.LogLevel.Information)\r\n .build();\r\n // when you get a message, process the message\r\n connection.on(\"newMessage\", handler);\r\n connection.onreconnecting(error => {\r\n console.assert(connection.state === signalR.HubConnectionState.Reconnecting);\r\n console.log(`Connection lost due to error \"${error}\". Reconnecting.`);\r\n });\r\n connection.onreconnected(connectionId => {\r\n console.assert(connection.state === signalR.HubConnectionState.Connected);\r\n console.log(`Connection reestablished. Connected with connectionId \"${connectionId}\".`);\r\n });\r\n // restart when you get a close event\r\n connection.onclose(async () => {\r\n console.log(`Connection closing. Attempting restart.`);\r\n await connection.start();\r\n });\r\n // start and display any caught exceptions in the console\r\n connection.start().catch(console.error);\r\n });\r\n }\r\n // start a sync cycle\r\n async startSync(instance: IPublicClientApplication, authorizedUser: User | null | undefined, config: SyncConfig | null | undefined): Promise<APIResult>\r\n {\r\n let result: APIResult = new APIResult();\r\n if (this.tenantNodes == null || this.tenantNodes.length == 0) {\r\n // we should not have an empty batch array for a test\r\n debugger;\r\n result.result = false;\r\n result.error = \"startSync: invalid parameters\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // execute post to reader endpoint\r\n result = await readerPost(instance, authorizedUser!, config!);\r\n return result;\r\n }\r\n}\r\nexport class TenantNode {\r\n expanded: boolean;\r\n status: string = \"\";\r\n name: string;\r\n tid: string;\r\n batchId: string;\r\n total: number = 0;\r\n read: number = 0;\r\n written: number = 0;\r\n deferred: number =0 ;\r\n nothingtosync: boolean;\r\n targets: TenantNode[];\r\n constructor(tid: string, name: string, batchId: string) {\r\n this.expanded = false;\r\n this.name = name;\r\n this.tid = tid;\r\n this.batchId = batchId;\r\n this.nothingtosync = false;\r\n this.targets = new Array<TenantNode>();\r\n this.update(0, 0, 0, 0);\r\n }\r\n update(total: number, read: number, written: number, deferred: number): void {\r\n this.total = total;\r\n this.read = read;\r\n this.written = written;\r\n this.deferred = deferred;\r\n if (this.read === 0 && this.written === 0) this.status = \"not started\";\r\n if (this.read > 0) {\r\n if (this.read < this.total) this.status = \"in progress\";\r\n else if (this.read === this.total) this.status = \"complete\";\r\n }\r\n else if (this.written > 0) {\r\n if (this.written + this.deferred < this.total) this.status = \"in progress\";\r\n else if (this.written === this.total) this.status = \"complete\";\r\n else if (this.written + this.deferred >= this.total) this.status = \"failed\";\r\n }\r\n }\r\n}\r\nexport class ResourceArray {\r\n resourceNodes: ResourceNode[];\r\n constructor(bInitialize: boolean, bClearLocalStorage: boolean) {\r\n this.resourceNodes = new Array<ResourceNode>();\r\n if (bInitialize) {\r\n this.init(bClearLocalStorage);\r\n }\r\n }\r\n // get resource data from localStorage or file\r\n init(bClearLocalStorage: boolean): void {\r\n console.log(`Calling ResourceArray::init(bClearLocalStorage: ${bClearLocalStorage ? \"true\" : \"false\"})`);\r\n // if we have a non-empty string value stored, read it from localStorage\r\n if (storageAvailable()) {\r\n let result = localStorage.getItem(\"ResourceArray\");\r\n if (result != null && typeof result === \"string\" && result !== \"\") {\r\n if (bClearLocalStorage) {\r\n localStorage.removeItem(\"ResourceArray\");\r\n }\r\n else {\r\n // read entire object from localstorage\r\n let raString: string = result;\r\n let resourceArray: ResourceArray = JSON.parse(raString);\r\n this.resourceNodes = resourceArray.resourceNodes;\r\n return;\r\n }\r\n }\r\n }\r\n // if storage unavailable or we were just asked to clear, read resources from file\r\n var resourceNodesString = JSON.stringify(resources);\r\n try {\r\n this.resourceNodes = deserializeArray(ResourceNode, resourceNodesString);\r\n } catch (e) {\r\n debugger;\r\n }\r\n }\r\n // read\r\n async read(instance: IPublicClientApplication, user: User): Promise<ResourceArray> {\r\n let resources: ResourceArray = new ResourceArray(false, false);\r\n resources.resourceNodes = await readResources(instance, user);\r\n return resources;\r\n }\r\n // save resource data to localstorage\r\n save(): void {\r\n // if we have localStorage, save resources\r\n if (storageAvailable()) {\r\n let raString: string = JSON.stringify(this);\r\n localStorage.setItem(\"ResourceArray\", raString);\r\n }\r\n }\r\n}\r\nexport class ResourceNode {\r\n type: string;\r\n resource: string;\r\n cost: number;\r\n expanded: boolean;\r\n resources: ResourceNode[];\r\n constructor(type: string, resource: string, cost: number) {\r\n this.type = type;\r\n this.resource = resource;\r\n this.cost = cost;\r\n this.expanded = false;\r\n this.resources = new Array<ResourceNode>();\r\n }\r\n}\r\nexport class ActorArray {\r\n actorNodes: ActorNode[];\r\n constructor(bClearLocalStorage: boolean) {\r\n this.actorNodes = new Array<ActorNode>();\r\n this.init(bClearLocalStorage);\r\n }\r\n // get initial data from localStorage or file\r\n init(bClearLocalStorage: boolean): void {\r\n console.log(`Calling ResourceArray::init(bClearLocalStorage: ${bClearLocalStorage ? \"true\" : \"false\"})`);\r\n // if we have a non-empty string value stored, read it from localStorage\r\n if (storageAvailable()) {\r\n let result = localStorage.getItem(\"RBACActors\");\r\n if (result != null && typeof result === \"string\" && result !== \"\") {\r\n if (bClearLocalStorage) {\r\n localStorage.removeItem(\"RBACActors\");\r\n }\r\n else {\r\n let actorArrayString: string = result;\r\n let aaFromLocalStorage: ActorArray = JSON.parse(actorArrayString);\r\n this.actorNodes = aaFromLocalStorage.actorNodes;\r\n return;\r\n }\r\n }\r\n }\r\n // if storage unavailable or we were just asked to clear, read defaults to enable usable UI\r\n var actorsString = JSON.stringify(actors);\r\n try {\r\n this.actorNodes = deserializeArray(ActorNode, actorsString);\r\n } catch (e) {\r\n debugger;\r\n }\r\n }\r\n}\r\nexport class ActorNode {\r\n type: string;\r\n actor: string;\r\n resource: string;\r\n role: string;\r\n updatedby: string;\r\n updatedon: string;\r\n actors: ActorNode[];\r\n constructor(type: string, actor: string, resource: string, role: string, updatedby: string, updatedon: string) {\r\n this.type = type;\r\n this.actor = actor;\r\n this.resource = resource;\r\n this.role = role;\r\n this.updatedby = updatedby;\r\n this.updatedon = updatedon;\r\n this.actors = new Array<ActorNode>();\r\n }\r\n}\r\n// ======================= Azure AD Graph API ===============================\r\n// helper functions\r\nfunction getGraphAPIScope(user: User): string {\r\n user = user;\r\n return \"Group.Read.All User.Read.All openid profile offline_access User.Read Contacts.Read CrossTenantInformation.ReadBasic.All\";\r\n}\r\n// TODO: this is where you want to trigger a re-authentication if token expires\r\nasync function graphDefineHeaders(\r\n instance: IPublicClientApplication,\r\n user: User\r\n): Promise<Headers> {\r\n const headers = new Headers();\r\n headers.append(\"Content-Type\", \"application/json\");\r\n headers.append(\"accept\", \"*/*\");\r\n const graphAPIScope: string = getGraphAPIScope(user);\r\n // only call acquireTokenByCode if we have never redeemed the code\r\n if (user.graphAccessToken == null || user.graphAccessToken === \"\") {\r\n try {\r\n let response: AuthenticationResult = await instance.acquireTokenByCode({\r\n code: user.spacode,\r\n });\r\n user.graphAccessToken = response.accessToken; // cache access token\r\n console.log(\"Front end token acquired by code: \" + user.graphAccessToken.slice(0, 20));\r\n }\r\n catch (error: any) {\r\n console.log(\"Front end token failure: \" + error);\r\n }\r\n }\r\n // otherwise, call acquireTokenSilent and deal with token expiration on exception\r\n else {\r\n try {\r\n let accounts: AccountInfo[] = instance.getAllAccounts();\r\n let homeAccountId = user.oid + \".\" + user.tid;\r\n let account: AccountInfo | undefined | null = null;\r\n for (let i: number = 0; i < accounts.length; i++) {\r\n if (accounts[i].homeAccountId == homeAccountId) {\r\n account = accounts[i];\r\n }\r\n }\r\n let response: AuthenticationResult = await instance.acquireTokenSilent({\r\n scopes: [graphAPIScope],\r\n account: account!\r\n });\r\n user.graphAccessToken = response.accessToken; // cache access token\r\n console.log(\"Front end token graph acquired silently: \" + user.graphAccessToken.slice(0, 20));\r\n }\r\n catch (error: any) {\r\n try {\r\n console.log(\"Front end graph token silent acquisition failure: \" + error);\r\n // fallback to redirect if silent acquisition fails\r\n let accounts: AccountInfo[] = instance.getAllAccounts();\r\n let homeAccountId = user.oid + \".\" + user.tid;\r\n let account: AccountInfo | null = null;\r\n for (let i: number = 0; i < accounts.length; i++) {\r\n if (accounts[i].homeAccountId == homeAccountId) {\r\n account = accounts[i];\r\n }\r\n }\r\n // assumption: this redirect will trigger login flow callbacks in program.cs\r\n instance.acquireTokenRedirect({\r\n scopes: [graphAPIScope],\r\n account: account!\r\n });\r\n }\r\n catch (error: any) {\r\n console.log(\"Front end graph token redirect acquisition failure: \" + error);\r\n }\r\n }\r\n }\r\n headers.append(\"Authorization\", `Bearer ${user.graphAccessToken}`);\r\n return headers;\r\n}\r\nexport async function groupsGet(instance: IPublicClientApplication, user: User | undefined, groupSearchString: string): Promise<{ groups: Group[], error: string }> {\r\n // need a logged in user to get graph users\r\n if (user == null || user.spacode == \"\") {\r\n return { groups: [], error: `500: invalid user passed to groupsGet` };\r\n }\r\n // create headers\r\n const headers = await graphDefineHeaders(instance, user);\r\n let options = { method: \"GET\", headers: headers };\r\n // make /groups endpoint call\r\n try {\r\n let groupsEndpoint: string = getGraphEndpoint(user.authority) + graphConfig.graphGroupsPredicate;\r\n groupsEndpoint += `/?$filter=startsWith(displayName, '${groupSearchString}')`;\r\n let response = await fetch(groupsEndpoint, options);\r\n let data = await response.json();\r\n if (typeof data.error !== \"undefined\") {\r\n return { groups: [], error: `${data.error.code}: ${data.error.message}` };\r\n }\r\n return { groups: data.value, error: `` };\r\n }\r\n catch (error: any) {\r\n console.log(error);\r\n return { groups: [], error: `Exception: ${error}` };\r\n }\r\n}\r\nexport async function oauth2PermissionGrantsGet(options: RequestInit, user: User, spid: string, oid: string): Promise<{ grants: string|null, id: string|null, error: string }> {\r\n try {\r\n // make /oauth2PermissionGrants endpoint call\r\n let spurl: string = getGraphEndpoint(user.authority) + graphConfig.graphOauth2PermissionGrantsPredicate;\r\n let url: URL = new URL(spurl);\r\n url.searchParams.append(\"$filter\", `resourceId eq '${spid}' and consentType eq 'Principal' and principalId eq '${oid}'`);\r\n let response = await fetch(url.href, options);\r\n let data = await response.json();\r\n if (typeof data.error != \"undefined\") {\r\n return { grants: null, id: null, error: `${data.error.code}: ${data.error.message}` };\r\n }\r\n // we assume there is only one such grant\r\n if (data.value.length != 1) {\r\n debugger;\r\n return { grants: null, id: null, error: `oauth2PermissionGrantsGet: more than one matching delegated consent grant.` };\r\n }\r\n return { grants: data.value[0].scope, id: data.value[0].id, error: `` };\r\n }\r\n catch (error: any) {\r\n console.log(error);\r\n return { grants: null, id: null, error: `Exception: ${error}` };\r\n }\r\n}\r\nexport async function oauth2PermissionGrantsSet(instance: IPublicClientApplication, loggedInUser: User, id: string, scopes: string): Promise<boolean> {\r\n // need a logged in user to get graph users\r\n if (loggedInUser == null || loggedInUser.spacode == \"\") {\r\n return false;\r\n }\r\n // make /oauth2PermissionGrants endpoint call\r\n try {\r\n let grantsurl: string = getGraphEndpoint(loggedInUser.authority);\r\n grantsurl += graphConfig.graphOauth2PermissionGrantsPredicate + `/${id}`;\r\n let scopesBody: string = `{ \"scope\": \"${scopes}\" }`;\r\n const headers = await graphDefineHeaders(instance, loggedInUser);\r\n let options: RequestInit = { method: \"PATCH\", headers: headers, body: scopesBody };\r\n let response = await fetch(grantsurl, options);\r\n let data = await response.json();\r\n if (response.status == 204 && response.statusText == \"No Content\") {\r\n return true;\r\n }\r\n else {\r\n debugger;\r\n console.log(`oauth2PermissionGrantsSet: PATCH failed ${data.error.code}: ${data.error.message}`);\r\n return false;\r\n }\r\n }\r\n catch (error: any) {\r\n debugger;\r\n console.log(error);\r\n return false;\r\n }\r\n}\r\nexport function requestAdminConsent(admin: User, tct: TenantConfigType): void {\r\n //\r\n // for app permissions (app roles) we must use the /.default scope for admin consent\r\n // https://learn.microsoft.com/EN-US/azure/active-directory/develop/v2-admin-consent#:~:text=In%20order%20to%20request%20app%20permissions%2C%20you%20must%20use%20the%20/.default%20value.\r\n // https://learn.microsoft.com/en-us/answers/questions/431784/how-to-grant-application-permissions-with-dynamic\r\n //\r\n // this means that, if we want to be granular about SyncReader vs. SyncWriter permissions, we must have separate Applications\r\n // in addition, if there are permissions like Group.Read.All, that also must be admin consented\r\n // for now, this function will perform admin consent for the Reader and Writer applications, we will come back to Group.Read.All and User.Read.All\r\n //\r\n // FYI - using the Challenge endpoint for app permissions caused the call to quietly fail without error\r\n // FYI - using the Challenge endpoint for this scope also caused the call to quietly fail withour error\r\n // scope = \"63100afe-506e-4bb2-8ff7-d8d5ab373129/.default\";\r\n //\r\n // thereforce, we are assuming that, for app permissions (app roles) or delegated permissions requiring admin consent, the Microsoft Identity Web Challenge endpoint does not work and we need to perform our own redirect to the admin consent endpoint\r\n // https://learn.microsoft.com/EN-US/azure/active-directory/develop/scopes-oidc#client-credentials-grant-flow-and-default\r\n // https://learn.microsoft.com/en-us/azure/active-directory/develop/v2-oauth2-client-creds-grant-flow#request-the-permissions-from-a-directory-admin\r\n //\r\n let adminConsentURL: string = getLoginEndpoint(admin.authority);\r\n adminConsentURL += admin.tid;\r\n adminConsentURL += \"/adminconsent\";\r\n let url: URL = new URL(adminConsentURL);\r\n let clientId: string = getAppId(admin.authority, tct);\r\n url.searchParams.append(\"client_id\", clientId);\r\n url.searchParams.append(\"redirect_uri\", window.location.origin);\r\n url.searchParams.append(\"domain_hint\", admin.companyDomain);\r\n url.searchParams.append(\"login_hint\", admin.mail);\r\n window.location.assign(url.href);\r\n}\r\nexport async function servicePrincipalGet(options: RequestInit, user: User, appid: string): Promise<{ spid: string, error: string }> {\r\n try {\r\n // make /servicePrincipals endpoint call to get the Service Principal ID\r\n let spurl: string = getGraphEndpoint(user.authority);\r\n spurl += graphConfig.graphServicePrincipalsPredicate;\r\n spurl += `(appId='${appid}')`;\r\n let url: URL = new URL(spurl);\r\n url.searchParams.append(\"$select\", \"id,appId,displayName\");\r\n let response = await fetch(url.href, options);\r\n let data = await response.json();\r\n if (typeof data.error !== \"undefined\") {\r\n return { spid: \"\", error: `${data.error.code}: ${data.error.message}` };\r\n }\r\n else {\r\n return { spid: data.id, error: `` };\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(error);\r\n return { spid: \"\", error: `Exception: ${error}` };\r\n }\r\n}\r\nexport async function signIn(user: User, tasks: TaskArray): Promise<boolean> {\r\n // admin authority is blank at signIn, lookup authority real-time\r\n if (user.authority == \"\") {\r\n // lookup authority for this user (the lookup call does it based on domain, but TID works as well to find authority)\r\n let tenant: Tenant = new Tenant();\r\n tenant.domain = user.tid;\r\n let bResult: boolean = await tenantUnauthenticatedLookup(tenant, false);\r\n if (bResult) {\r\n // success, we now know cloud instance where this tenant is provisioned\r\n user.authority = tenant.authority;\r\n }\r\n else {\r\n // can't sign in without an authority\r\n debugger;\r\n return false;\r\n }\r\n }\r\n // SignIn by an admin consents the app, Challenge adds incremental permissions dynamically, but requires a consented app - TODO: TEST FRESH CONSENT EXPERIENCE\r\n let signinURL: string = window.location.href;\r\n switch (user.authority) {\r\n case graphConfig.authorityWW:\r\n signinURL += \"MicrosoftIdentity/Account/SignIn\";\r\n break;\r\n case graphConfig.authorityUS:\r\n signinURL += \"USGov/SignIn\";\r\n break;\r\n case graphConfig.authorityCN:\r\n signinURL += \"China/SignIn\";\r\n break;\r\n default:\r\n // unknown authority\r\n return false;\r\n }\r\n let url: URL = new URL(signinURL);\r\n url.searchParams.append(\"redirectUri\", window.location.origin);\r\n if (user.oid !== \"1\") {\r\n url.searchParams.append(\"loginHint\", user.mail);\r\n const regex = /@([a-zA-Z0-9.-]+\\.[a-zA-Z]{2,})$/;\r\n const regexMatch = user.mail.match(regex);\r\n let domain: string = regexMatch ? regexMatch[1] : \"organizations\";\r\n url.searchParams.append(\"domainHint\", domain);\r\n }\r\n else {\r\n // \"1\" is the dummy OID set when initializing the application from JSON. No need to provide any specific hint.\r\n // This means that a user has not yet been specified by the admin. If it had been, oid would not be \"1\".\r\n url.searchParams.append(\"domainHint\", \"organizations\");\r\n }\r\n tasks.setTaskStart(\"initialization\", new Date());\r\n tasks.setTaskStart(\"authenticate user\", new Date());\r\n window.location.assign(url.href);\r\n return true;\r\n}\r\nexport function signInIncrementally(user: User, scope: string): void {\r\n debugger;\r\n if (user.oid == \"1\") return;\r\n // for dynamic delegated permissions, we can use the Microsoft Identity Web Account Controller Challenge method\r\n let challengeURL: string = window.location.href;\r\n switch (user.authority) {\r\n case graphConfig.authorityWW:\r\n challengeURL += \"MicrosoftIdentity/Account/Challenge\";\r\n break;\r\n case graphConfig.authorityUS:\r\n challengeURL += \"USGov/Challenge\";\r\n break;\r\n case graphConfig.authorityCN:\r\n challengeURL += \"China/Challenge\";\r\n break;\r\n default:\r\n // unknown authority\r\n return;\r\n }\r\n let url: URL = new URL(challengeURL);\r\n url.searchParams.append(\"redirectUri\", window.location.origin);\r\n url.searchParams.append(\"scope\", scope);\r\n url.searchParams.append(\"domainHint\", \"organizations\");\r\n url.searchParams.append(\"loginHint\", user.mail);\r\n window.location.assign(url.href);\r\n}\r\nexport async function signOut(user: User): Promise<boolean>{\r\n if (user.oid == \"1\") return false;\r\n // set logout_hint in the .NET session for streamlined logout\r\n let userEndpoint: string = window.location.href;\r\n userEndpoint += \"user\";\r\n let url = new URL(userEndpoint);\r\n url.searchParams.append(\"oid\", user.oid);\r\n url.searchParams.append(\"tid\", user.tid);\r\n url.searchParams.append(\"loginHint\", user.loginHint);\r\n url.searchParams.append(\"verb\", \"LOGOUT\");\r\n let options = { method: \"PATCH\" };\r\n let userLogoutResponse: Response = await fetch(url.href, options);\r\n if (userLogoutResponse.status == 200 && userLogoutResponse.statusText == \"OK\") {\r\n console.log(`Successfully set admin ${user.mail} logout_hint`);\r\n }\r\n else {\r\n console.log(`Failed to set admin ${user.mail} logout_hint`);\r\n return false;\r\n }\r\n // start the logout process triggering callbacks during logout\r\n // OnRedirectToIdentityProviderForSignOut - this is where we set the logout_hint for user we are trying to logout\r\n // OnSignedOutCallbackRedirect - called when the call sucessfully completes\r\n let signoutURL: string = window.location.href;\r\n switch (user.authority) {\r\n case graphConfig.authorityWW:\r\n signoutURL += \"MicrosoftIdentity/Account/SignOut\";\r\n break;\r\n case graphConfig.authorityUS:\r\n signoutURL += \"USGov/SignOut\";\r\n break;\r\n case graphConfig.authorityCN:\r\n signoutURL += \"China/SignOut\";\r\n break;\r\n default:\r\n // unknown authority\r\n return false;\r\n }\r\n url = new URL(signoutURL);\r\n url.searchParams.append(\"redirectUri\", window.location.origin);\r\n window.location.assign(url.href);\r\n return true;\r\n}\r\n//tenantRelationshipsGetByDomain - query AAD for associated company name and id\r\nexport async function tenantRelationshipsGetByDomain(loggedInUser: User, tenant: Tenant, instance: IPublicClientApplication, debug: boolean): Promise<boolean> {\r\n if (debug) debugger;\r\n // do we already have a valid tenant name? if so, nothing to add\r\n if (tenant.name != null && tenant.name !== \"\") return false;\r\n // if needed, retrieve and cache access token\r\n if (loggedInUser.graphAccessToken != null && loggedInUser.graphAccessToken === \"\") {\r\n console.log(`tenantRelationshipsGetByDomain called with invalid logged in user: ${loggedInUser.name}`);\r\n try {\r\n let response: AuthenticationResult = await instance.acquireTokenByCode({ code: loggedInUser.spacode, scopes: [\"user.read\", \"contacts.read\", \"CrossTenantInformation.ReadBasic.All\"] });\r\n loggedInUser.graphAccessToken = response.accessToken; // cache access token on the user\r\n console.log(\"tenantRelationshipsGetByDomain: Front end token acquired: \" + loggedInUser.graphAccessToken.slice(0, 20));\r\n }\r\n catch (error: any) {\r\n console.log(\"tenantRelationshipsGetByDomain: Front end token failure: \" + error);\r\n return false; // failed to get access token, no need to re-render\r\n }\r\n }\r\n // prepare Authorization headers as part of options\r\n const headers = new Headers();\r\n const bearer = `Bearer ${loggedInUser.graphAccessToken}`;\r\n headers.append(\"Authorization\", bearer);\r\n let options = { method: \"GET\", headers: headers };\r\n // make tenant endpoint call\r\n try {\r\n // create tenant info endpoint\r\n var tenantEndpoint = getGraphEndpoint(tenant.authority) + graphConfig.graphTenantByDomainPredicate;\r\n tenantEndpoint += \"(domainName='\";\r\n tenantEndpoint += tenant.domain;\r\n tenantEndpoint += \"')\";\r\n console.log(\"tenantRelationshipsGetByDomain: Attempting GET from /findTenantInformationByDomainName:\", tenantEndpoint);\r\n let response = await fetch(tenantEndpoint, options);\r\n // status IS 200, but statusText no longer returns \"OK\" 1/26/2025\r\n if (response.status == 200) { // && response.statusText == \"OK\") {\r\n let data = await response.json();\r\n if (data) {\r\n if (data.error != null) {\r\n debugger;\r\n console.log(\"tenantRelationshipsGetByDomain: Failed GET from /findTenantInformationByDomainName: \", data.error.message);\r\n return false;\r\n }\r\n else if (data.displayName != null && data.displayName !== \"\") {\r\n // set domain information on passed tenant\r\n tenant.tid = data.tenantId;\r\n tenant.name = data.displayName;\r\n console.log(\"tenantRelationshipsGetByDomain: Successful GET from /findTenantInformationByDomainName: \", data.displayName);\r\n return true; // success, need UX to re-render\r\n }\r\n }\r\n else {\r\n console.log(\"tenantRelationshipsGetByDomain: Failed to GET from /findTenantInformationByDomainName: \", tenantEndpoint);\r\n }\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(\"Failed to GET from /findTenantInformationByDomainName: \", error);\r\n return false; // failed, no need for UX to re-render\r\n }\r\n return false; // failed, no need for UX to re-render\r\n}\r\n//tenantRelationshipsGetById - query AAD for associated company name and domain\r\nexport async function tenantRelationshipsGetById(loggedInUser: User, tenant: Tenant, ii: InitInfo, instance: IPublicClientApplication, debug: boolean): Promise<boolean> {\r\n ii = ii;\r\n console.log(\"**** tenantRelationshipsGetById\");\r\n if (debug) debugger;\r\n // if needed, retrieve and cache access token\r\n if (loggedInUser.graphAccessToken === \"\") {\r\n try {\r\n let response: AuthenticationResult = await instance.acquireTokenByCode({ code: loggedInUser.spacode, scopes: [\"user.read\", \"contacts.read\", \"CrossTenantInformation.ReadBasic.All\"] });\r\n loggedInUser.graphAccessToken = response.accessToken; // cache access token\r\n console.log(\"tenantRelationshipsGetById: Front end token acquired: \" + loggedInUser.graphAccessToken.slice(0, 20));\r\n }\r\n catch (error: any) {\r\n console.log(\"tenantRelationshipsGetById: Front end token failure: \" + error);\r\n return false; // failed to get access token, no need to re-render\r\n }\r\n }\r\n // prepare Authorization headers as part of options\r\n const headers = new Headers();\r\n const bearer = `Bearer ${loggedInUser.graphAccessToken}`;\r\n headers.append(\"Authorization\", bearer);\r\n let options = { method: \"GET\", headers: headers };\r\n // make tenant endpoint call\r\n try {\r\n // create tenant info endpoint\r\n var tenantEndpoint = getGraphEndpoint(loggedInUser.authority) + graphConfig.graphTenantByIdPredicate;\r\n tenantEndpoint += \"(tenantId='\";\r\n tenantEndpoint += tenant.tid;\r\n tenantEndpoint += \"')\";\r\n console.log(\"tenantRelationshipsGetById: Attempting GET from /findTenantInformationByTenantId:\", tenantEndpoint);\r\n let response = await fetch(tenantEndpoint, options);\r\n let data = await response.json();\r\n if (data && typeof data.displayName !== undefined && data.displayName !== \"\") {\r\n // set domain information on tenant\r\n if (tenant !== undefined) {\r\n tenant.name = data.displayName;\r\n tenant.domain = data.defaultDomainName;\r\n }\r\n else {\r\n console.log(\"tenantRelationshipsGetById: missing associated tenant for logged in user.\");\r\n debugger;\r\n }\r\n console.log(\"tenantRelationshipsGetById: Successful GET from /findTenantInformationByTenantId: \", data.displayName);\r\n return true; // success, need UX to re-render\r\n }\r\n else {\r\n console.log(\"tenantRelationshipsGetById: Failed to GET from /findTenantInformationByTenantId: \", tenantEndpoint);\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(\"tenantRelationshipsGetById: Failed to GET from /findTenantInformationByTenantId: \", error);\r\n return false; // failed, no need for UX to re-render\r\n }\r\n return false; // failed, no need for UX to re-render\r\n}\r\n//tenantUnauthenticatedLookup (from https://gettenantpartitionweb.azurewebsites.net/js/gettenantpartition.js)\r\nexport async function tenantUnauthenticatedLookup(tenant: Tenant, debug: boolean): Promise<boolean> {\r\n if (debug) debugger;\r\n // prepare the 3 endpoints and corresponding regular expressions\r\n let endpoints: string[] = [graphConfig.authorityWW, graphConfig.authorityUS, graphConfig.authorityCN];\r\n let regexes: RegExp[] = [graphConfig.authorityWWRegex, graphConfig.authorityUSRegex, graphConfig.authorityCNRegex];\r\n // make unauthenticated well-known openid endpoint call(s)\r\n let response = null;\r\n try {\r\n for (let j = 0; j < 3; j++) {\r\n // create well-known openid endpoint\r\n var openidEndpoint = endpoints[j];\r\n if (tenant.lookupfield == \"Domain\") {\r\n openidEndpoint += tenant.domain;\r\n }\r\n else if (tenant.lookupfield == \"Tenant ID\") {\r\n openidEndpoint += tenant.tid;\r\n }\r\n openidEndpoint += \"/.well-known/openid-configuration\";\r\n console.log(\"Attempting GET from openid well-known endpoint: \", openidEndpoint);\r\n response = await fetch(openidEndpoint);\r\n if (response.status == 200) {\r\n let data = await response.json();\r\n if (data) {\r\n // store tenant ID and authority\r\n var tenantAuthEndpoint = data.authorization_endpoint;\r\n // USGov/China tenants are registered in WW with USGov/China authority values!\r\n for (let k = 0; k < 3; k++) {\r\n var authMatches = tenantAuthEndpoint.match(regexes[k]);\r\n if (authMatches != null) {\r\n tenant.tid = authMatches[2];\r\n tenant.authority = authMatches[1];\r\n console.log(`Successful GET from openid well-known endpoint: tid: ${tenant.tid} authority: ${tenant.authority}`);\r\n return true; // success, need UX to re-render\r\n }\r\n }\r\n }\r\n else {\r\n console.log(`Failed JSON parse of openid well-known endpoint response ${openidEndpoint}.`);\r\n }\r\n }\r\n else {\r\n console.log(`Failed GET from ${openidEndpoint}.`);\r\n }\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(\"Failed to GET from openid well-known endpoint: \", error);\r\n }\r\n if (tenant.tid == \"\" || tenant.authority == \"\") {\r\n console.log(`GET from openid well-known endpoint failed to find tenant: ${response ? response.statusText : \"unknown\"}`);\r\n }\r\n return false; // failed, no need for UX to re-render\r\n}\r\nexport async function userDelegatedScopesGet(instance: IPublicClientApplication, loggedInUser: User, tenant: Tenant): Promise<{ scopes: string|null, id: string|null, error: string }> {\r\n // need a logged in user and valid tenant to query graph\r\n if (loggedInUser == null || loggedInUser.spacode == \"\" || tenant == null) {\r\n debugger;\r\n return { scopes: null, id: null, error: `500: invalid parameter(s) passed to getUserDelegatedScopes` };\r\n }\r\n // create headers\r\n const headers = await graphDefineHeaders(instance, loggedInUser);\r\n let options: RequestInit = { method: \"GET\", headers: headers };\r\n try {\r\n // first, cache Graph resource ID (service principal) for this tenant if we don't have it already\r\n if (tenant.graphSP == \"\") {\r\n let { spid, error } = await servicePrincipalGet(options, loggedInUser, \"00000003-0000-0000-c000-000000000000\");\r\n if (error != \"\") {\r\n debugger;\r\n return { scopes: null, id: null, error: `${error}` };\r\n }\r\n tenant.graphSP = spid;\r\n }\r\n // then, retrieve the delegated Graph permissions assigned to this user\r\n let { grants, id, error } = await oauth2PermissionGrantsGet(options, loggedInUser, tenant.graphSP, loggedInUser.oid);\r\n if (error != \"\") {\r\n debugger;\r\n return { scopes: null, id: null, error: `${error}` };\r\n }\r\n return { scopes: grants, id: id, error: `` };\r\n }\r\n catch (error: any) {\r\n debugger;\r\n console.log(error);\r\n return { scopes: null, id: null, error: `Exception: ${error}` };\r\n }\r\n}\r\nexport async function userDelegatedScopesRemove(instance: IPublicClientApplication, loggedInUser: User, tenant: Tenant, scope: string): Promise<boolean> {\r\n // need a logged in user and valid tenant to query graph\r\n if (loggedInUser == null || loggedInUser.spacode == \"\" || tenant == null) {\r\n debugger;\r\n return false;\r\n }\r\n // get current set of delegated scopes in order to remove passed scope\r\n let { scopes, id, error } = await userDelegatedScopesGet(instance, loggedInUser, tenant);\r\n if (error != \"\") {\r\n debugger;\r\n console.log(`userDelegatedScopesRemove: cannot find userDelegatedScopes for ${loggedInUser.mail}: ${error}`);\r\n return false;\r\n }\r\n // remove passed scope (case sensitive)\r\n scopes = scopes!.replace(scope, \"\");\r\n // set updated oauth2permissions\r\n let removed: boolean = await oauth2PermissionGrantsSet(instance, loggedInUser, id!, scopes);\r\n if (!removed) {\r\n debugger;\r\n console.log(`userDelegatedScopesRemove: cannot set oauth2PermissionGrants for ${loggedInUser.mail}: ${error}`);\r\n return false;\r\n }\r\n // replace scope array on logged in user\r\n loggedInUser.scopes = scopes.split(\" \");\r\n return removed;\r\n}\r\n//usersGet - GET from AAD Users endpoint\r\nexport async function usersGet(instance: IPublicClientApplication, user: User | undefined): Promise<{ users: string[], error: string }> {\r\n // need a logged in user to get graph users\r\n if (user == null || user.spacode == \"\") {\r\n return { users: [], error: `500: invalid user passed to usersGet` };\r\n }\r\n // make /users endpoint call\r\n try {\r\n // create headers\r\n const headers = await graphDefineHeaders(instance, user);\r\n let options = { method: \"GET\", headers: headers };\r\n let usersEndpoint = getGraphEndpoint(user.authority);\r\n usersEndpoint += graphConfig.graphUsersPredicate;\r\n let response = await fetch(usersEndpoint, options);\r\n let data = await response.json();\r\n if (typeof data.error !== \"undefined\") {\r\n return { users: [], error: `${data.error.code}: ${data.error.message}` };\r\n }\r\n let users = new Array<string>();\r\n for (let user of data.value) {\r\n users.push(user.mail);\r\n }\r\n return { users: users, error: `` };\r\n }\r\n catch (error: any) {\r\n console.log(error);\r\n return { users: [], error: `Exception: ${error}` };\r\n }\r\n}\r\n// ======================= Mindline SyncConfig API ===============================\r\nexport async function auditConfigAdd(instance: IPublicClientApplication, user: User, ac: AuditConfig, debug: boolean): Promise<APIResult> {\r\n return auditConfigPost(instance, user, ac, debug);\r\n}\r\nexport async function auditConfigRetrieve(instance: IPublicClientApplication, user: User, configurationId: string, debug: boolean): Promise<APIResult> {\r\n return auditConfigGet(instance, user, configurationId, debug);\r\n}\r\nexport async function auditConfigEdit(instance: IPublicClientApplication, user: User, configurationId: string, config: AuditConfig, debug: boolean): Promise<APIResult> {\r\n return auditConfigPut(instance, user, configurationId, config, debug);\r\n}\r\nexport async function auditConfigRemove(instance: IPublicClientApplication, user: User, configurationId: string, debug: boolean): Promise<APIResult> {\r\n return auditConfigDelete(instance, user, configurationId, debug);\r\n}\r\nexport async function auditEventsRetrieve(instance: IPublicClientApplication, authorizedUser: User, debug: boolean): Promise<APIResult> {\r\n return auditEventsGet(instance, authorizedUser, debug);\r\n}\r\nexport async function configEdit(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n config: SyncConfig,\r\n setConfigId: (id: string) => void,\r\n setSelectedConfigs: (selectedConfigs: { [id: string]: boolean | number[] }) => void,\r\n workspace: Workspace,\r\n ii: InitInfo,\r\n debug: boolean): Promise<APIResult>\r\n{\r\n let result: APIResult = new APIResult();\r\n if (config.id === \"1\") {\r\n result = await configPost(instance, authorizedUser, config, workspace.id, debug);\r\n if (result.result) {\r\n // config id was updated from \"1\"\r\n if (setConfigId) {\r\n setConfigId(config.id);\r\n }\r\n // set this config as the selected config\r\n const newSelection = {};\r\n Object.defineProperty(newSelection, config.id, { value: true, writable: true, enumerable: true });\r\n setSelectedConfigs(newSelection);\r\n // update associated configs in this workspace\r\n let idx: number = workspace.associatedConfigs.findIndex((id: string) => id == \"1\");\r\n if (idx !== -1) {\r\n workspace.associatedConfigs.splice(idx, 1);\r\n workspace.associatedConfigs.push(config.id);\r\n }\r\n }\r\n }\r\n else {\r\n result = await configPut(instance, authorizedUser, config, debug);\r\n }\r\n ii.save();\r\n return result;\r\n}\r\nexport async function configEnable(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n configurationId: string,\r\n enabled: boolean,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n result = await configPatch(instance, authorizedUser, configurationId, enabled, debug);\r\n return result;\r\n}\r\nexport async function configConsentForRead(instance: IPublicClientApplication, authorizedUser: User, configId: string, tid: string, consent: boolean): Promise<APIResult> {\r\n return configConsentReadPut(instance, authorizedUser, configId, tid, consent);\r\n}\r\nexport async function configConsentForWrite(instance: IPublicClientApplication, authorizedUser: User, configId: string, tid: string, consent: boolean): Promise<APIResult> {\r\n return configConsentWritePut(instance, authorizedUser, configId, tid, consent);\r\n}\r\nexport async function configsRefresh(instance: IPublicClientApplication, authorizedUser: User, workspaceId: string, ii: InitInfo, debug: boolean): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (debug) debugger;\r\n try {\r\n let workspace: Workspace | undefined = ii.ws.find((w) => w.id === workspaceId);\r\n if (workspace != null) {\r\n // clear SyncConfig associations as we are about to reset\r\n workspace.associatedConfigs.length = 0;\r\n // GET configs associated with this workspace\r\n let result: APIResult = await workspaceConfigsGet(instance, authorizedUser, workspace.id, debug);\r\n if (!result.result) return result;\r\n // process returned workspace components\r\n let configSelectedId: string = \"\";\r\n for (let cid of workspace.associatedConfigs) {\r\n let config = ii.cs.find((c: SyncConfig) => c.id === cid);\r\n if (config != null && config.sel) {\r\n ii.cs.find((c: SyncConfig) => c.id === cid) ??\r\n ii.acs.find((a: AuditConfig) => a.id === cid);\r\n\r\n if (config && config.sel) {\r\n configSelectedId = config.id;\r\n }\r\n }\r\n\r\n // Pass combined list to processor\r\n processReturnedConfigs(workspace, ii, result.array!, configSelectedId);\r\n // tag components with workspaceIDs\r\n ii.tagWithWorkspaces();\r\n }\r\n return result;\r\n } else {\r\n // workspace not found\r\n result.result = false;\r\n result.error = \"Workspace not found.\";\r\n result.status = 404;\r\n return result;\r\n }\r\n\r\n } catch (error: any) {\r\n console.log(error.message);\r\n result.error = error.message;\r\n result.result = false;\r\n result.status = 500;\r\n return result;\r\n }\r\n}\r\n\r\nexport async function configRemove(instance: IPublicClientApplication, authorizedUser: User, config: SyncConfig, workspaceId: string, debug: boolean): Promise<APIResult> {\r\n return configDelete(instance, authorizedUser, config, workspaceId, debug);\r\n}\r\nexport async function initGet(instance: IPublicClientApplication, user: User, ii: InitInfo, tasks: TaskArray, debug: boolean): Promise<APIResult> {\r\n console.log(`>>>>>> initGet`);\r\n let result: APIResult = new APIResult();\r\n if (debug)\r\n debugger;\r\n // lookup authority for this user (the lookup call does it based on domain, but TID works as well to find authority)\r\n let tenant: Tenant = new Tenant();\r\n tenant.tid = user.tid;\r\n tenant.domain = user.tid;\r\n let bResult: boolean = await tenantUnauthenticatedLookup(tenant, debug);\r\n if (bResult) {\r\n // success, we now know cloud instance where this tenant is provisioned\r\n user.authority = tenant.authority;\r\n // get tenant name and domain from AAD to pass to Configuration API\r\n tasks.setTaskStart(\"GET tenant details\", new Date());\r\n result.result = await tenantRelationshipsGetById(user, tenant, ii, instance, debug);\r\n tasks.setTaskEnd(\"GET tenant details\", new Date(), \"complete\");\r\n // store tenant name and domain on user, POST user and tenant to back end\r\n if (result.result) {\r\n user.companyName = tenant.name;\r\n user.companyDomain = tenant.domain;\r\n tasks.setTaskStart(\"POST config init\", new Date());\r\n result = await initPost(instance, user, debug);\r\n tasks.setTaskEnd(\"POST config init\", new Date(), result.result ? \"complete\" : \"failed\");\r\n }\r\n // simlarly, if we just did our first post, then query config backend for workspace(s) associated with this user\r\n if (result.result) {\r\n tasks.setTaskStart(\"GET workspaces\", new Date());\r\n result = await workspaceInfoGet(instance, user, ii, debug);\r\n tasks.setTaskEnd(\"GET workspaces\", new Date(), result.result ? \"complete\" : \"failed\");\r\n }\r\n console.log(\"initGet complete. Version: \" + getSyncVersion());\r\n return result;\r\n }\r\n else {\r\n result.error = `Failed to retrieve authority for user \"${user.mail}\" TID ${user.tid}.`;\r\n result.result = false;\r\n return result;\r\n }\r\n}\r\nexport async function tenantAdd(instance: IPublicClientApplication, authorizedUser: User, tenant: Tenant, workspaceId: string): Promise<APIResult> {\r\n return tenantPost(instance, authorizedUser, tenant, workspaceId);\r\n}\r\nexport async function tenantRemove(instance: IPublicClientApplication, authorizedUser: User, tenant: Tenant, workspaceId: string, debug: boolean): Promise<APIResult> {\r\n return tenantDelete(instance, authorizedUser, tenant, workspaceId, debug);\r\n}\r\nexport async function userAdd(instance: IPublicClientApplication, authorizedUser: User, user: User, workspaceId: string): Promise<APIResult> {\r\n return adminPost(instance, authorizedUser, user, workspaceId);\r\n}\r\nexport async function userRemove(instance: IPublicClientApplication, authorizedUser: User, user: User, workspaceId: string): Promise<APIResult> {\r\n return adminDelete(instance, authorizedUser, user, workspaceId);\r\n}\r\nexport async function workspaceEdit(instance: IPublicClientApplication, authorizedUser: User, workspaceId: string, workspaceName: string): Promise<APIResult>\r\n{\r\n return await workspacePut(instance, authorizedUser, workspaceId, workspaceName);\r\n}\r\n// retrieve Workspace(s), User(s), Tenant(s), SyncConfig(s) given newly logged in user\r\nfunction processReturnedAdmins(workspace: Workspace, ii: InitInfo, returnedAdmins: Array<Object>, adminSelectedId: string) {\r\n returnedAdmins.map((item: any) => {\r\n // are we already tracking this user?\r\n let user: User | null | undefined = null;\r\n let usIndex = ii.us.findIndex((u) => (u.oid === item.userId || u.oid === item.email));\r\n if (usIndex === -1) {\r\n // start tracking\r\n let dummyIndex = ii.us.findIndex((u) => u.oid === \"1\");\r\n if (dummyIndex !== -1) {\r\n // clear and overwrite dummy\r\n user = ii.us.at(dummyIndex);\r\n // replace dummy User oid \"1\" with real oid in associatedTenants of this workspace\r\n let idx: number = workspace.associatedUsers.findIndex((oid: string) => oid == \"1\");\r\n if (idx !== -1) {\r\n workspace.associatedUsers.splice(idx, 1);\r\n workspace.associatedUsers.push(item.userId);\r\n }\r\n }\r\n else {\r\n // create and track new user\r\n user = new User();\r\n ii.us.push(user);\r\n }\r\n } else {\r\n // already tracking this user\r\n user = ii.us.at(usIndex);\r\n }\r\n // restore selection\r\n user!.sel = (adminSelectedId === item.userId);\r\n // refresh all the data available from the server\r\n user!.oid = item.userId ? item.userId : item.email;\r\n user!.name = item.firstName ?? user!.name;\r\n user!.mail = item.email;\r\n user!.tid = item.tenantId;\r\n // *try* to set authority/companyName/companyDomain from tenant returned in previous call processReturnedTenants (it may not be there)\r\n // ASSUMPTION: in terms of setting authority, user either comes from\r\n // 1. .NET session, in which case user has an authority from the token\r\n // 2. SyncConfig API, in which case either\r\n // a. associated tenant with authority exists for this user (i.e. workspace owner has called POST init/configuration and created Admin and Tenant at same time)\r\n // b. user has never logged in (i.e. workspace owner has invited an incomplete admin to the workspace and there is no corresponding tenant)\r\n // i. this means that we *may* not have an authority stored for this user\r\n let tenant: Tenant | undefined = ii.ts.find((t) => (t.tid === user!.tid));\r\n if (tenant != undefined) {\r\n user!.authority = tenant.authority;\r\n user!.companyName = tenant.name;\r\n user!.companyDomain = tenant.domain;\r\n }\r\n // ensure this workspace tracks this user\r\n let idx = workspace.associatedUsers.findIndex((u) => u === user!.oid);\r\n if (idx == -1) workspace.associatedUsers.push(user!.oid);\r\n });\r\n ii.save();\r\n}\r\nfunction processReturnedTenants(workspace: Workspace, ii: InitInfo, returnedTenants: Array<Object>, tenantSelectedId: string) {\r\n returnedTenants.map((item: any) => {\r\n // are we already tracking this tenant?\r\n let tenant: Tenant | null | undefined = null;\r\n let tsIndex = ii.ts.findIndex((t) => t.tid === item.tenantId);\r\n if (tsIndex === -1) {\r\n // start tracking\r\n let dummyIndex = ii.ts.findIndex((t) => t.tid === \"1\");\r\n if (dummyIndex !== -1) {\r\n // clear and overwrite dummy\r\n tenant = ii.ts.at(dummyIndex);\r\n // replace dummy Tenant id \"1\" with real id in associatedTenants of this workspace\r\n let idx: number = workspace.associatedTenants.findIndex((tid: string) => tid == \"1\");\r\n if (idx !== -1) {\r\n workspace.associatedTenants.splice(idx, 1);\r\n workspace.associatedTenants.push(item.tenantId);\r\n }\r\n } else {\r\n // create and track new tenant\r\n tenant = new Tenant();\r\n ii.ts.push(tenant);\r\n }\r\n } else {\r\n // already tracking this tenant\r\n tenant = ii.ts.at(tsIndex);\r\n }\r\n // restore selection\r\n tenant!.sel = (tenantSelectedId === item.tenantId);\r\n // refresh all the data available from the server\r\n tenant!.tid = item.tenantId;\r\n tenant!.name = item.name;\r\n tenant!.domain = item.domain;\r\n tenant!.tenantType = item.type.toLowerCase(); // should now be strings\r\n\r\n // canonicalize authority when getting it from config backend\r\n const regex = /^(https:\\/\\/login.microsoftonline.(?:us|com)\\/)organizations\\/v2.0$/;\r\n const regexMatch = item.authority.match(regex);\r\n tenant!.authority = regexMatch ? regexMatch[1] : item.authority;\r\n\r\n // ensure this workspace tracks this tenant\r\n let idx = workspace.associatedTenants.findIndex((t) => t === item.tenantId);\r\n if (idx == -1) workspace.associatedTenants.push(item.tenantId);\r\n });\r\n ii.save();\r\n}\r\nfunction processReturnedConfigs(workspace: Workspace, ii: InitInfo, returnedConfigs: Array<Object>, configSelectedId: string) {\r\n // process returned configs\r\n returnedConfigs.map((item: any) => {\r\n // are we already tracking this config?\r\n let config: SyncConfig | null | undefined = null;\r\n let csIndex = ii.cs.findIndex((c) => c.id === item.id);\r\n if (csIndex === -1) {\r\n // start tracking\r\n let dummyIndex = ii.cs.findIndex((c) => c.id === \"1\");\r\n if (dummyIndex !== -1) {\r\n // clear and overwrite dummy\r\n config = ii.cs.at(dummyIndex);\r\n // replace dummy SyncConfig id \"1\" with real id in associatedConfigs of this workspace\r\n let idx: number = workspace.associatedConfigs.findIndex((id: string) => id == \"1\");\r\n if (idx !== -1) {\r\n workspace.associatedConfigs.splice(idx, 1);\r\n workspace.associatedConfigs.push(item.id);\r\n }\r\n } else {\r\n // create and track new workspace\r\n config = new SyncConfig();\r\n ii.cs.push(config);\r\n }\r\n } else {\r\n // already tracking this config\r\n config = ii.cs.at(csIndex);\r\n }\r\n // restore selection\r\n config!.sel = (configSelectedId === item.id);\r\n // refresh all the data available from the server\r\n config!.id = item.id;\r\n config!.workspaceId = item.workspaceId;\r\n config!.name = item.name;\r\n config!.description = item.description;\r\n config!.isEnabled = item.isEnabled;\r\n // create TenantConfigInfo array\r\n config!.tenants.length = 0;\r\n item.tenants.map((tci: any) => {\r\n let tenantConfigInfo = new TenantConfigInfo();\r\n tenantConfigInfo.tid = tci.tenantId;\r\n tenantConfigInfo.sourceGroupId = tci.sourceGroupId ?? \"\";\r\n tenantConfigInfo.sourceGroupName = tci.sourceGroupName ?? \"\";\r\n tenantConfigInfo.targetGroupId = tci.targetGroupId ?? \"\";\r\n tenantConfigInfo.targetGroupName = tci.targetGroupName ?? \"\";\r\n tenantConfigInfo.configurationTenantType = tci.configurationTenantType.toLowerCase();\r\n tenantConfigInfo.deltaToken = tci.deltaToken ?? \"\";\r\n tenantConfigInfo.configId = config!.id;\r\n tenantConfigInfo.batchId = tci.batchId ?? \"\";\r\n tenantConfigInfo.isReadPermissionConsented = tci.isReadPermissionConsented;\r\n tenantConfigInfo.isWritePermissionConsented = tci.isWritePermissionConsented;\r\n config!.tenants.push(tenantConfigInfo);\r\n });\r\n // ensure this workspace tracks this config\r\n let idx = workspace.associatedConfigs.findIndex((c) => c === item.id);\r\n if (idx == -1) workspace.associatedConfigs.push(item.id);\r\n });\r\n ii.save();\r\n}\r\nasync function workspaceInfoGet(instance: IPublicClientApplication, user: User, ii: InitInfo, debug: boolean): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (debug) debugger;\r\n try {\r\n result = await workspacesGet(instance, user, debug);\r\n if (result.result) {\r\n for (let o of result.array! as any[]) {\r\n // are we already tracking this workspace?\r\n let workspace: Workspace | null | undefined = null;\r\n let wsIndex = ii.ws.findIndex((w) => w.id === o.id);\r\n if (wsIndex === -1) {\r\n // start tracking\r\n let dummyIndex = ii.ws.findIndex((w) => w.id === \"1\");\r\n if (dummyIndex !== -1) {\r\n // clear and overwrite dummy\r\n workspace = ii.ws.at(dummyIndex);\r\n }\r\n else {\r\n // create and track new workspace\r\n workspace = new Workspace();\r\n ii.ws.push(workspace);\r\n }\r\n } else {\r\n // already tracking this workspace\r\n workspace = ii.ws.at(wsIndex);\r\n }\r\n // preserve selected admin, tenant, and config\r\n let adminSelectedId: string = \"\";\r\n for (let oid of workspace!.associatedUsers) {\r\n let user = ii.us.find((u: User) => u.oid === oid);\r\n if (user != null && user.sel) {\r\n adminSelectedId = user.oid;\r\n }\r\n }\r\n let tenantSelectedId: string = \"\";\r\n for (let tid of workspace!.associatedTenants) {\r\n let tenant = ii.ts.find((t: Tenant) => t.tid === tid);\r\n if (tenant != null && tenant.sel) {\r\n tenantSelectedId = tenant.tid;\r\n }\r\n }\r\n let configSelectedId: string = \"\";\r\n for (let cid of workspace!.associatedConfigs) {\r\n let config = ii.cs.find((c: SyncConfig) => c.id === cid);\r\n if (config != null && config.sel) {\r\n configSelectedId = config.id;\r\n }\r\n }\r\n // clear associations as we are about to reset\r\n workspace!.associatedUsers.length = 0;\r\n workspace!.associatedTenants.length = 0;\r\n workspace!.associatedConfigs.length = 0;\r\n // set id and name based on returned data\r\n workspace!.id = o.id;\r\n workspace!.name = o.name;\r\n workspace!.ownerid = o.workspaceOwnerUserId;\r\n // parallel GET admins, tenants, configs associated with this workspace\r\n let adminsPromise: Promise<APIResult> = adminsGet(instance, user, workspace!.id, debug);\r\n let tenantsPromise: Promise<APIResult> = tenantsGet(instance, user, workspace!.id, debug);\r\n let configsPromise: Promise<APIResult> = configsGet(instance, user, workspace!.id, debug);\r\n // wait for all to finish, return on any failure\r\n let [adminsResult, tenantsResult, configsResult] = await Promise.all([adminsPromise, tenantsPromise, configsPromise]);\r\n if (!adminsResult.result) return adminsResult;\r\n if (!tenantsResult.result) return tenantsResult;\r\n if (!configsResult.result) return configsResult;\r\n // process returned workspace components (tenants first as they have the authorities that admins depend on)\r\n processReturnedTenants(workspace!, ii, tenantsResult.array!, tenantSelectedId);\r\n processReturnedAdmins(workspace!, ii, adminsResult.array!, adminSelectedId);\r\n processReturnedConfigs(workspace!, ii, configsResult.array!, configSelectedId);\r\n // tag components with workspaceIDs\r\n ii.tagWithWorkspaces();\r\n }\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(error.message);\r\n result.error = error.message;\r\n }\r\n result.result = false;\r\n result.status = 500;\r\n return result;\r\n}\r\n// ======================= PowerBI Access Token Retrieval ===============================\r\nexport async function getPowerBIAccessToken(\r\n instance: IPublicClientApplication,\r\n user: User\r\n): Promise<string> {\r\n let accesstoken: string = \"\";\r\n let syncversion: string = getSyncVersion();\r\n // retrieve access token silently\r\n try {\r\n let accounts: AccountInfo[] = instance.getAllAccounts();\r\n let homeAccountId = user.oid + \".\" + user.tid;\r\n let account: AccountInfo | null = null;\r\n for (let i: number = 0; i < accounts.length; i++) {\r\n if (accounts[i].homeAccountId == homeAccountId) {\r\n account = accounts[i];\r\n }\r\n }\r\n let response: AuthenticationResult = await instance.acquireTokenSilent({\r\n scopes: [\"https://analysis.windows.net/powerbi/api/App.Read.All\",\r\n \"https://analysis.windows.net/powerbi/api/Dataset.ReadWrite.All\",\r\n \"https://analysis.windows.net/powerbi/api/Report.ReadWrite.All\",\r\n \"https://analysis.windows.net/powerbi/api/Workspace.ReadWrite.All\"\r\n ],\r\n account: account!\r\n });\r\n accesstoken = response.accessToken; // cache access token\r\n console.log(\"PowerBI token acquired silently: \" + accesstoken.slice(0, 20));\r\n }\r\n catch (error: any) {\r\n console.log(\"PowerBI token failed to be acquired silently\");\r\n }\r\n console.log(syncversion);\r\n return accesstoken;\r\n}\r\n// ======================= Azure REST API ===============================\r\n// TODO: this is where you want to trigger a re-authentication if token expires\r\nasync function azureDefineHeaders(\r\n instance: IPublicClientApplication,\r\n user: User\r\n): Promise<Headers> {\r\n const headers = new Headers();\r\n headers.append(\"Content-Type\", \"application/json\");\r\n headers.append(\"accept\", \"*/*\");\r\n // authorization header - if needed, retrieve and cache access token\r\n if (user.azureAccessToken == null || user.azureAccessToken === \"\") {\r\n try {\r\n let accounts: AccountInfo[] = instance.getAllAccounts();\r\n let homeAccountId = user.oid + \".\" + user.tid;\r\n let account: AccountInfo | null = null;\r\n for (let i: number = 0; i < accounts.length; i++) {\r\n if (accounts[i].homeAccountId == homeAccountId) {\r\n account = accounts[i];\r\n }\r\n }\r\n let response: AuthenticationResult = await instance.acquireTokenSilent({\r\n scopes: [\"https://management.azure.com/user_impersonation\"],\r\n account: account!\r\n });\r\n user.azureAccessToken = response.accessToken; // cache access token\r\n console.log(\"Front end token acquired silently: \" + user.azureAccessToken.slice(0, 20));\r\n }\r\n catch (error: any) {\r\n try {\r\n console.log(\"Front end token silent acquisition failure: \" + error);\r\n // fallback to redirect if silent acquisition fails\r\n let accounts: AccountInfo[] = instance.getAllAccounts();\r\n let homeAccountId = user.oid + \".\" + user.tid;\r\n let account: AccountInfo | null = null;\r\n for (let i: number = 0; i < accounts.length; i++) {\r\n if (accounts[i].homeAccountId == homeAccountId) {\r\n account = accounts[i];\r\n }\r\n }\r\n instance.acquireTokenRedirect({\r\n scopes: [\"https://management.azure.com/user_impersonation\"],\r\n account: account!\r\n });\r\n }\r\n catch (error: any) {\r\n console.log(\"Front end token popup acquisition failure: \" + error);\r\n }\r\n }\r\n }\r\n headers.append(\"Authorization\", `Bearer ${user.azureAccessToken}`);\r\n return headers;\r\n}\r\nexport async function canListRootAssignments(instance: IPublicClientApplication, user: User): Promise<boolean> {\r\n // need a logged in user to call Azure REST API\r\n if (user == null || user.spacode == \"\") {\r\n return false;\r\n }\r\n // make Azure REST API call\r\n try {\r\n // create headers\r\n const headers = await azureDefineHeaders(instance, user);\r\n let options = { method: \"GET\", headers: headers };\r\n let listrootassignmentsEndpoint: string = azureConfig.azureListRootAssignments;\r\n listrootassignmentsEndpoint += \"'\";\r\n listrootassignmentsEndpoint += user.oid;\r\n listrootassignmentsEndpoint += \"'\";\r\n let response = await fetch(listrootassignmentsEndpoint, options);\r\n if (response.status == 200) {\r\n let data: any = await response.json();\r\n data = data;\r\n debugger;\r\n console.log(\"Successful call to Azure Resource Graph list root assignments\");\r\n }\r\n else {\r\n console.log(await processErrors(response));\r\n return false;\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(error);\r\n return false;\r\n }\r\n return true;\r\n}\r\nexport async function elevateGlobalAdminToUserAccessAdmin(instance: IPublicClientApplication, user: User): Promise<boolean> {\r\n // need a logged in user to call Azure REST API\r\n if (user == null || user.spacode == \"\") {\r\n return false;\r\n }\r\n // make Azure REST API call\r\n try {\r\n // create headers\r\n const headers = await azureDefineHeaders(instance, user);\r\n let options = { method: \"POST\", headers: headers };\r\n let elevateaccessEndpoint: string = azureConfig.azureElevateAccess;\r\n let response = await fetch(elevateaccessEndpoint, options);\r\n if (response.status == 200) {\r\n console.log(\"Successful call to Azure Resource Graph list root assignments\");\r\n }\r\n else {\r\n console.log(await processErrors(response));\r\n return false;\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(error);\r\n return false;\r\n }\r\n return true; \r\n}\r\nasync function readResources(instance: IPublicClientApplication, user: User): Promise<ResourceNode[]> {\r\n // need a logged in user to call Azure REST API\r\n let resources: ResourceNode[] = new Array<ResourceNode>();\r\n if (user == null || user.spacode == \"\") {\r\n return resources;\r\n }\r\n // make Azure REST API call\r\n try {\r\n // create headers\r\n const headers = await azureDefineHeaders(instance, user);\r\n let options = { method: \"GET\", headers: headers };\r\n let listrootassignmentsEndpoint: string = azureConfig.azureListRootAssignments;\r\n listrootassignmentsEndpoint += \"'\";\r\n listrootassignmentsEndpoint += user.oid;\r\n listrootassignmentsEndpoint += \"'\";\r\n let response = await fetch(listrootassignmentsEndpoint, options);\r\n if (response.status == 200) {\r\n let data = await response.json();\r\n data = data;\r\n debugger;\r\n console.log(\"Successful call to Azure Resource Graph list root assignments\");\r\n }\r\n else {\r\n console.log(await processErrors(response));\r\n return resources;\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(error);\r\n return resources;\r\n }\r\n return resources;\r\n}\r\n// ======================= HYBRIDSPA.TS -- Mindline SyncConfig API helper functions ===============================\r\nfunction getAPIScope(user: User): string {\r\n let apiAppID: string = \"8d95d21c-c378-4bb0-9f52-88c30d271e7a\";\r\n let authority: string = user.authority.toLowerCase();\r\n if (authority.startsWith(\"https://login.microsoftonline.com/\"))\r\n apiAppID = \"8d95d21c-c378-4bb0-9f52-88c30d271e7a\";\r\n else if (authority.startsWith(\"https://login.microsoftonline.us/\"))\r\n apiAppID = \"48da942e-ea3d-49e4-a054-81649012f8f2\";\r\n else if (authority.startsWith(\"https://login.partner.microsoftonline.cn/\"))\r\n apiAppID = \"c91d32e4-dcc5-4d77-826a-16e93ffce666\";\r\n let apiScope: string = `api://${apiAppID}/Config.Write`;\r\n return apiScope;\r\n}\r\n// TODO: this is where you want to trigger a re-authentication if token expires\r\nasync function mindlineDefineHeaders(\r\n instance: IPublicClientApplication,\r\n user: User\r\n): Promise<Headers> {\r\n const headers = new Headers();\r\n headers.append(\"Content-Type\", \"application/json\");\r\n headers.append(\"accept\", \"*/*\");\r\n // always call acquireTokenSilent, handling expired tokens on exception\r\n const apiScope: string = getAPIScope(user);\r\n try {\r\n let accounts: AccountInfo[] = instance.getAllAccounts();\r\n let homeAccountId = user.oid + \".\" + user.tid;\r\n let account: AccountInfo | null = null;\r\n for (let i: number = 0; i < accounts.length; i++) {\r\n if (accounts[i].homeAccountId == homeAccountId) {\r\n account = accounts[i];\r\n }\r\n }\r\n let response: AuthenticationResult = await instance.acquireTokenSilent({\r\n scopes: [apiScope],\r\n account: account!\r\n });\r\n user.mindlineAccessToken = response.accessToken; // cache access token\r\n console.log(\"Front end mindline token acquired silently: \" + user.mindlineAccessToken.slice(0, 20));\r\n }\r\n catch (error: any) {\r\n try {\r\n console.log(\"Front end mindline token silent acquisition failure, triggering redirect: \" + error);\r\n // fallback to redirect if silent acquisition fails\r\n let accounts: AccountInfo[] = instance.getAllAccounts();\r\n let homeAccountId = user.oid + \".\" + user.tid;\r\n let account: AccountInfo | null = null;\r\n for (let i: number = 0; i < accounts.length; i++) {\r\n if (accounts[i].homeAccountId == homeAccountId) {\r\n account = accounts[i];\r\n }\r\n }\r\n // assumption: this redirect will trigger login flow callbacks in program.cs\r\n instance.acquireTokenRedirect({\r\n scopes: [apiScope],\r\n account: account!\r\n });\r\n }\r\n catch (error: any) {\r\n console.log(\"Front end mindline token redirect acquisition failure: \" + error);\r\n }\r\n }\r\n headers.append(\"Authorization\", `Bearer ${user.mindlineAccessToken}`);\r\n return headers;\r\n}\r\nexport async function processErrors(response: Response): Promise<string> {\r\n let errorString: string = \"\";\r\n if (response.status === 401) {\r\n errorString = response.statusText;\r\n if (errorString != \"\") return errorString;\r\n }\r\n let data = await response.json();\r\n // process errors from Mindline SyncConfig API\r\n if (data.error !== undefined) {\r\n errorString = `Error: ${data.error} Message: ${data.message}`;\r\n } else if (data.errors !== undefined) {\r\n let errorArray = Object.keys(data.errors);\r\n let errorlist: string = \"\";\r\n errorString = errorArray.reduce(\r\n (acc, curr) => acc + curr + \": \" + data.errors[curr] + \" \",\r\n errorlist\r\n );\r\n } else if (data.title !== undefined) {\r\n errorString = data.title;\r\n } else {\r\n debugger;\r\n }\r\n return errorString;\r\n}\r\n//adminDelete\r\nexport async function adminDelete(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n user: User,\r\n workspaceId: string\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we need either oid or mail for the user and valid workspace id\r\n if ((user.oid == \"\" && user.mail == \"\") || workspaceId == \"\") {\r\n result.result = false;\r\n result.error = \"adminDelete: invalid parameters\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // define admin endpoint and add workspaceId parameter\r\n let url: URL | null = null;\r\n url = new URL(mindlineConfig.adminEndpoint());\r\n url.searchParams.append(\"workspaceId\", workspaceId);\r\n url.searchParams.append(\"email\", user.mail);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // make endpoint call\r\n let options = { method: \"DELETE\", headers: headers };\r\n try {\r\n console.log(\"Attempting DELETE from /admin: \" + url!.href);\r\n let response = await fetch(url!.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful DELETE from /admin: ${url!.href}`);\r\n return result;\r\n } else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed DELETE from /admin: ${url.href}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n } catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//adminsGet\r\nexport async function adminsGet(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n workspaceID: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we need a workspace id\r\n if (workspaceID === \"\") {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"adminsGet: no workspace provided\";\r\n return result;\r\n }\r\n // create endpoint\r\n let endpoint: string = mindlineConfig.adminsEndpoint();\r\n // add parameter to endpoint\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"workspaceId\", workspaceID);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n // make endpoint call\r\n let options = { method: \"GET\", headers: headers };\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting GET from /admins: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let returnedArray: Array<Object> = await response.json();\r\n if (returnedArray != null) {\r\n result.array = returnedArray;\r\n let initialValue: string = \"\";\r\n console.log(`Successful GET from /admins: ${result.array.reduce((acc, curr: any) => acc + curr.email + \" \", initialValue)}`);\r\n return result;\r\n }\r\n else {\r\n result.error = `Failed GET from /admins: failed to JSON-parse response`;\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n else {\r\n console.log(`Failed GET from /admins: ${url.href}`);\r\n result.error = await processErrors(response);\r\n result.status = 500;\r\n result.result = false;\r\n console.log(result.error);\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//adminPost: write validated admin to back end\r\nexport async function adminPost(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n user: User,\r\n workspaceId: string\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (user.mail == \"\" || user.authority == \"\" || user.tid === \"\") {\r\n result.result = false;\r\n result.error = \"adminPost: invalid argument\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create admin endpoint\r\n let endpoint: string = mindlineConfig.adminEndpoint();\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // create admin body\r\n let adminBody: string = `\r\n {\"email\": \"${user.mail}\", \r\n \"tenantId\": \"${user.tid}\", \r\n \"workspaceId\": \"${workspaceId}\"\r\n }`;\r\n let options = { method: \"POST\", headers: headers, body: adminBody };\r\n // make admin endpoint call\r\n try {\r\n console.log(\"Attempting POST to /admin: \" + endpoint);\r\n let response = await fetch(endpoint, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful POST to /admin: ${adminBody}`);\r\n return result;\r\n } else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed POST to /admin: ${adminBody}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n } catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n\r\n//auditConfigPost: write audit config to back end\r\nexport async function auditConfigPost(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n config: AuditConfig,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n instance = instance;\r\n user = user;\r\n debug = debug;\r\n let result: APIResult = new APIResult();\r\n if (config.id === \"\") {\r\n result.result = false;\r\n result.error = \"auditConfigPost: invalid config ID\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create no parameter audit config post endpoint\r\n let endpoint: string = mindlineConfig.auditConfigEndpoint();\r\n // create config headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n // create config body\r\n let configBody: string = `\r\n {\r\n \"workspaceId\": \"${config.workspaceId}\",\r\n \"name\": \"${config.name}\",\r\n \"description\": \"${config.description}\",\r\n \"isEnabled\": ${config.isEnabled},\r\n \"email\": \"${config.email}\",\r\n \"tenantId\": \"${config.tenantId}\"\r\n }`;\r\n let options = { method: \"POST\", headers: headers, body: configBody };\r\n // make config endpoint call\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting POST to /audit-configuration: \" + endpoint);\r\n let response = await fetch(endpoint, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let data = await response.json();\r\n config.id = data;\r\n console.log(\r\n `Successful AuditConfigID: ${data} from POST to /audit-configuration: ${configBody}`\r\n );\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed POST to /audit-configuration: ${configBody}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.status = 500;\r\n result.result = false;\r\n result.error = error.message;\r\n console.log(result.error);\r\n return result;\r\n }\r\n return result;\r\n}\r\n\r\n//auditConfigGET: get audit config from back end\r\nexport async function auditConfigGet(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n configurationId: string,\r\n debug: boolean): Promise<APIResult> {\r\n instance = instance;\r\n user = user;\r\n debug = debug;\r\n\r\n let result: APIResult = new APIResult();\r\n\r\n // Validate user\r\n if (user.mail == undefined || user.mail == \"\") {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"auditConfigGet: no user provided\";\r\n return result;\r\n }\r\n\r\n //Validate input\r\n if (!configurationId || configurationId.trim() === \"\") {\r\n result.result = false;\r\n result.error = \"auditConfigGet: configurationId is required\";\r\n result.status = 500;\r\n return result;\r\n }\r\n\r\n //Define endpoint with query parameter (configurationId)\r\n let endpoint: string = mindlineConfig.auditConfigEndpoint();\r\n let url: URL = new URL(`${endpoint}/${configurationId}`);\r\n\r\n //Create headers with Entra ID token\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n\r\n let options = { method: \"GET\", headers: headers };\r\n\r\n //Make the GET request\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting GET to /audit-configuration: \" + url.href);\r\n\r\n const response = await fetch(url.href, options);\r\n\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let data = await response.json();\r\n result.result = true;\r\n result.status = response.status;\r\n console.log(\"Successful GET from /audit-configuration:\", data);\r\n } else {\r\n result.error = await processErrors(response);\r\n console.log(\"Failed GET from /audit-configuration: \" + url.href);\r\n console.log(result.error);\r\n result.status = response.status;\r\n result.result = false;\r\n }\r\n } catch (error: any) {\r\n result.status = 500;\r\n result.result = false;\r\n result.error = error.message;\r\n console.log(\"Error during auditConfigGet:\", error.message);\r\n }\r\n\r\n return result;\r\n}\r\n\r\n//auditConfigPUT: put audit config from back end \r\nexport async function auditConfigPut(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n configurationId: string,\r\n config: AuditConfig, // model with name, description, isEnabled, isReadPermissionConsented, email\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n\r\n // Validate configurationId\r\n if (!configurationId || configurationId.trim() === \"\") {\r\n result.result = false;\r\n result.error = \"auditConfigPut: invalid configuration ID\";\r\n result.status = 500;\r\n return result;\r\n }\r\n\r\n // Build endpoint with query parameter\r\n let endpoint: string = mindlineConfig.auditConfigEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"configurationId\", configurationId);\r\n\r\n // Create headers with token\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n\r\n // Build body \r\n let configBody: string = `\r\n {\r\n \"name\": \"${config.name}\",\r\n \"description\": \"${config.description}\",\r\n \"isEnabled\": ${config.isEnabled},\r\n \"isReadPermissionConsented\": ${config.isReadPermissionConsented},\r\n \"email\": \"${config.email}\"\r\n }`;\r\n\r\n let options = { method: \"PUT\", headers: headers, body: configBody };\r\n\r\n // Call API\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting PUT to /audit-configuration: \" + url.href);\r\n\r\n const response = await fetch(url.href, options);\r\n\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n result.result = true;\r\n console.log(`Successful PUT to ${url.href}: ${configBody}`);\r\n } else {\r\n result.error = await processErrors(response);\r\n console.error(`Failed PUT to ${url.href}: ${configBody}`);\r\n result.status = response.status;\r\n result.result = false;\r\n }\r\n } catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.error(\"Error during auditConfigPut:\", error.message);\r\n }\r\n\r\n return result;\r\n}\r\n\r\n//auditConfigDELETE: delete audit config from back end \r\nexport async function auditConfigDelete(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n configurationId: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n\r\n // Validate user\r\n if (!user.mail) {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"auditConfigDelete: no user provided\";\r\n return result;\r\n }\r\n\r\n // Validate configurationId\r\n if (!configurationId || configurationId.trim() === \"\") {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"auditConfigDelete: configurationId is required\";\r\n return result;\r\n }\r\n\r\n // Build endpoint with query parameter\r\n let endpoint: string = mindlineConfig.auditConfigEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"configurationId\", configurationId);\r\n\r\n // Build request headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n const options = { method: \"DELETE\", headers: headers };\r\n\r\n try {\r\n if (debug) debugger;\r\n console.log(`Attempting DELETE to /audit-configuration: ${url.href}`);\r\n\r\n const response = await fetch(url.href, options);\r\n\r\n if (response.ok) {\r\n result.result = true;\r\n console.log(`Successful DELETE for configurationId ${configurationId}`);\r\n } else {\r\n result.error = await processErrors(response);\r\n result.status = response.status;\r\n result.result = false;\r\n console.error(`Failed DELETE for configurationId ${configurationId}`);\r\n }\r\n } catch (error: any) {\r\n result.status = 500;\r\n result.result = false;\r\n result.error = error.message;\r\n console.error(\"Error during auditConfigDelete:\", error.message);\r\n }\r\n\r\n return result;\r\n}\r\n\r\n\r\nexport async function auditEventsGet(instance: IPublicClientApplication, user: User, debug: boolean): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we need a valid user\r\n if (user.mail == undefined || user.mail == \"\") {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"auditEventsGet: no user provided\";\r\n return result;\r\n }\r\n // create endpoint\r\n let endpoint: string = mindlineConfig.auditEventsEndpoint();\r\n let url: URL = new URL(endpoint);\r\n // create workspace headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n // make workspace endpoint call\r\n let options = { method: \"GET\", headers: headers };\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting GET from /audit-events endpoint: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let json = await response.json();\r\n if (Array.isArray(json.data)) {\r\n let returnedArray: Array<Object> = json.data;\r\n if (returnedArray != null) {\r\n result.array = returnedArray;\r\n let initialValue: string = \"\";\r\n console.log(`Successful GET from /audit-events: ${result.array.reduce((acc, curr: any) => acc + curr.initiatedByUser + \" \", initialValue)}`);\r\n return result;\r\n }\r\n }\r\n result.error = `Failed GET from /audit-events: failed to JSON-parse response`;\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n else {\r\n console.log(`Failed GET from /audit-events: ${url.href}`);\r\n result.error = await processErrors(response);\r\n result.status = 500;\r\n result.result = false;\r\n console.log(result.error);\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//configConsentReadPut\r\nexport async function configConsentReadPut(instance: IPublicClientApplication, authorizedUser: User, configId: string, tid: string, consent: boolean): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // create parameterized config consent endpoint\r\n let endpoint: string = mindlineConfig.configConsentEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"configurationId\", configId);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // create body\r\n let configConsentReadBody: string = `\r\n {\r\n \"tenantId\": \"${tid}\",\r\n \"isReadPermissionConsented\": ${consent ? \"true\" : \"false\"}\r\n }`;\r\n // make endpoint call\r\n let options = { method: \"PUT\", headers: headers, body: configConsentReadBody };\r\n try {\r\n console.log(\"Attempting PUT read consent to /configuration/consent: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful PUT to ${url.href}`);\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed PUT to ${url.href}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//configConsentWritePut\r\nexport async function configConsentWritePut(instance: IPublicClientApplication, authorizedUser: User, configId: string, tid: string, consent: boolean): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // create parameterized config consent endpoint\r\n let endpoint: string = mindlineConfig.configConsentEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"configurationId\", configId);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // create body\r\n let configConsentWriteBody: string = `\r\n {\r\n \"tenantId\": \"${tid}\",\r\n \"isWritePermissionConsented\": ${consent ? \"true\" : \"false\"}\r\n }`;\r\n // make endpoint call\r\n let options = { method: \"PUT\", headers: headers, body: configConsentWriteBody };\r\n try {\r\n console.log(\"Attempting PUT read consent to /configuration/consent: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful PUT to ${url.href}`);\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed PUT to ${url.href}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//configDelete\r\nexport async function configDelete(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n config: SyncConfig,\r\n workspaceId: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n debug = debug;\r\n let result: APIResult = new APIResult();\r\n if (config.id === \"\" || workspaceId == \"\") {\r\n result.result = false;\r\n result.error = \"configPost: invalid config ID\";\r\n result.status = 500;\r\n return result;\r\n }\r\n let url: URL | null = null;\r\n url = new URL(mindlineConfig.configEndpoint());\r\n url.searchParams.append(\"configurationId\", config.id);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // make endpoint call\r\n let options = { method: \"DELETE\", headers: headers };\r\n try {\r\n console.log(\"Attempting DELETE from /config: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful DELETE from /config`);\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed DELETE from ${url.href}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//configPatch\r\nexport async function configPatch(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n configurationId: string,\r\n enabled: boolean,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (configurationId === \"\") {\r\n result.result = false;\r\n result.error = \"configPatch: invalid config ID\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create parametrized config endpoint\r\n let endpoint: string = mindlineConfig.configEnabledEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"configurationId\", configurationId);\r\n url.searchParams.append(\"isEnabled\", enabled.toString());\r\n // create config headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n let options = { method: \"PATCH\", headers: headers };\r\n // make config endpoint call\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting PATCH to /config: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful PATCH to ${url.href}: ${enabled.toString()}`);\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed PATCH to ${url.href}: ${enabled.toString()}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//configPost\r\nexport async function configPost(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n config: SyncConfig,\r\n workspaceId: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (config.id === \"\") {\r\n result.result = false;\r\n result.error = \"configPost: invalid config ID\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create no parameter config endpoint\r\n let endpoint: string = mindlineConfig.configEndpoint();\r\n // create config headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // create config body\r\n let configBody: string = `\r\n {\r\n \"workspaceId\": \"${workspaceId}\",\r\n \"name\": \"${config.name}\",\r\n \"description\": \"${config.description}\",\r\n \"isEnabled\": ${config.isEnabled},\r\n \"tenants\": [`;\r\n config.tenants.map((tci) => {\r\n // be sure we send null and not \"null\" in body\r\n let sourceGroupId: string = tci.sourceGroupId != \"\" ? `\"${tci.sourceGroupId}\"` : \"null\";\r\n let sourceGroupName: string = tci.sourceGroupName != \"\" ? `\"${tci.sourceGroupName}\"` : \"null\";\r\n let targetGroupId: string = tci.targetGroupId != \"\" ? `\"${tci.targetGroupId}\"` : \"null\";\r\n let targetGroupName: string = tci.targetGroupName != \"\" ? `\"${tci.targetGroupName}\"` : \"null\";\r\n // if last character is } we need a comma first\r\n let needComma: boolean = configBody.slice(-1) === \"}\";\r\n if (needComma) configBody += \",\";\r\n configBody += `{\r\n \"tenantId\": \"${tci.tid}\",\r\n \"sourceGroupId\": ${sourceGroupId},\r\n \"sourceGroupName\": ${sourceGroupName},\r\n \"targetGroupId\": ${targetGroupId},\r\n \"targetGroupName\": ${targetGroupName},\r\n \"configurationTenantType\": \"${tci.configurationTenantType}\"\r\n }`;\r\n });\r\n configBody += `]}`;\r\n let options = { method: \"POST\", headers: headers, body: configBody };\r\n // make config endpoint call\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting POST to /config: \" + endpoint);\r\n let response = await fetch(endpoint, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let data = await response.json();\r\n config.id = data;\r\n console.log(\r\n `Successful ConfigID: ${data} from POST to /config: ${configBody}`\r\n );\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed PUT to /config: ${configBody}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.status = 500;\r\n result.result = false;\r\n result.error = error.message;\r\n console.log(result.error);\r\n return result;\r\n }\r\n}\r\n//configPut\r\nexport async function configPut(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n config: SyncConfig,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (config.id === \"\") {\r\n result.result = false;\r\n result.error = \"configPost: invalid config ID\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create parametrized config endpoint\r\n let endpoint: string = mindlineConfig.configEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"configurationId\", config.id);\r\n // create config headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // create config body\r\n let configBody: string = `\r\n {\r\n \"name\": \"${config.name}\",\r\n \"description\": \"${config.description}\",\r\n \"tenants\": [`;\r\n config.tenants.map((tci: TenantConfigInfo) => {\r\n // if last character is } we need a comma first\r\n let needComma: boolean = configBody.slice(-1) === \"}\";\r\n if (needComma) configBody += \",\";\r\n // be sure we send null and not \"null\" in body\r\n let sourceGroupId: string = tci.sourceGroupId != \"\" ? `\"${tci.sourceGroupId}\"` : \"null\";\r\n let sourceGroupName: string = tci.sourceGroupName != \"\" ? `\"${tci.sourceGroupName}\"` : \"null\";\r\n let targetGroupId: string = tci.targetGroupId != \"\" ? `\"${tci.targetGroupId}\"` : \"null\";\r\n let targetGroupName: string = tci.targetGroupName != \"\" ? `\"${tci.targetGroupName}\"` : \"null\";\r\n configBody += `{\r\n \"tenantId\": \"${tci.tid}\",\r\n \"sourceGroupId\": ${sourceGroupId},\r\n \"sourceGroupName\": ${sourceGroupName},\r\n \"targetGroupId\": ${targetGroupId},\r\n \"targetGroupName\": ${targetGroupName},\r\n \"configurationTenantType\": \"${tci.configurationTenantType}\",\r\n \"deltaToken\": \"${tci.deltaToken}\"\r\n }`;\r\n });\r\n configBody += `]}`;\r\n let options = { method: \"PUT\", headers: headers, body: configBody };\r\n // make config endpoint call\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting PUT to /config: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful PUT to ${url.href}: ${configBody}`);\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed PUT to ${url.href}: ${configBody}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//configsGet\r\nexport async function configsGet(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n workspaceID: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we need a workspace id\r\n if (workspaceID === \"\") {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"configsGet: no workspace provided\";\r\n return result;\r\n }\r\n // create endpoint\r\n let endpoint: string = mindlineConfig.configsEndpoint();\r\n // add parameter to endpoint\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"workspaceId\", workspaceID);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n // make endpoint call\r\n let options = { method: \"GET\", headers: headers };\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting GET from /configurations: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let returnedArray: Array<Object> = await response.json();\r\n if (returnedArray != null) {\r\n result.array = returnedArray;\r\n let initialValue: string = \"\";\r\n console.log(`Successful GET from /configurations: ${result.array.reduce((acc, curr: any) => acc + curr.name + \" \", initialValue)}`);\r\n return result;\r\n }\r\n else {\r\n result.error = `Failed GET from /configurations: failed to JSON-parse response`;\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n else {\r\n console.log(`Failed GET from /configurations: ${url.href}`);\r\n result.error = await processErrors(response);\r\n result.status = 500;\r\n result.result = false;\r\n console.log(result.error);\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//workspaceConfigsGET\r\nexport async function workspaceConfigsGet(\r\n instance: IPublicClientApplication,\r\n user: User | null | undefined,\r\n workspaceId: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n\r\n if (!workspaceId) {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"workspaceConfigsGet: no workspace provided\";\r\n return result;\r\n }\r\n\r\n // Create endpoint\r\n let endpoint: string = mindlineConfig.workspaceConfigsEndpoint();\r\n // Add parameter to endpoint\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"workspaceId\", workspaceId);\r\n\r\n // Headers with Bearer token\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n\r\n let options = { method: \"GET\", headers };\r\n\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting GET from /workspaceConfigs: \" + url.href);\r\n\r\n const response = await fetch(url.href, options);\r\n\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n const returnedArray: Array<any> = await response.json();\r\n\r\n if (returnedArray) {\r\n // Filter by configurationType (sync or auditLog)\r\n result.array = returnedArray.map((config) => ({\r\n id: config.id,\r\n name: config.name,\r\n description: config.description,\r\n configurationType: config.configurationType,\r\n isEnabled: config.isEnabled,\r\n }));\r\n\r\n console.log(\r\n `workspaceConfigsGet: found ${result.array.length} configs.` \r\n );\r\n return result;\r\n } else {\r\n result.error =\r\n \"workspaceConfigsGet: failed to parse returned configs\";\r\n result.result = false;\r\n result.status = 500;\r\n }\r\n } else {\r\n result.error = await processErrors(response);\r\n result.result = false;\r\n result.status = response.status;\r\n console.log(result.error);\r\n }\r\n } catch (error: any) {\r\n result.error = error.message;\r\n result.result = false;\r\n result.status = 500;\r\n }\r\n\r\n return result;\r\n}\r\n//initPost\r\nexport async function initPost(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we expect valid company name and domain by this point\r\n if (user.companyName === \"\" || user.companyDomain === \"\") {\r\n result.result = false;\r\n result.error = \"initPost: invalid company name or domain\"\r\n result.status = 500;\r\n return result;\r\n }\r\n // create init endpoint\r\n let endpoint: string = mindlineConfig.initEndpoint();\r\n // create init headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n // create init body\r\n let initBody: string = `\r\n {\r\n \"tenantCreateModel\": {\r\n \"tenantId\": \"${user.tid}\",\r\n \"name\": \"${user.companyName}\",\r\n \"domain\": \"${user.companyDomain}\",\r\n \"type\": \"aad\",\r\n \"authority\": \"${user.authority}\"\r\n }\r\n }`;\r\n let options = { method: \"POST\", headers: headers, body: initBody };\r\n // make init endpoint call\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting POST to /configuration/init: \" + endpoint);\r\n let response = await fetch(endpoint, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful POST to /configuration/init: ${initBody}`);\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n result.status = 500;\r\n result.result = false;\r\n console.log(`Failed POST to /configuration/init: ${initBody}`);\r\n console.log(result.error);\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n console.log(result.error);\r\n }\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n}\r\n//tenantDelete\r\nexport async function tenantDelete(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n tenant: Tenant,\r\n workspaceId: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n debug = debug;\r\n let result: APIResult = new APIResult();\r\n // we expect valid tid amd workspaceId\r\n if (tenant.tid === \"\" || workspaceId === \"\") {\r\n result.result = false;\r\n result.error = \"tenantDelete: invalid tid, workspaceId\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create parametrized tenant endpoint\r\n let url: URL = new URL(mindlineConfig.tenantEndpoint());\r\n url.searchParams.append(\"tenantId\", tenant.tid);\r\n url.searchParams.append(\"workspaceId\", workspaceId);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // make tenant endpoint call\r\n let options = { method: \"DELETE\", headers: headers };\r\n try {\r\n console.log(\"Attempting DELETE from /tenant: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful DELETE from /tenant: ${url.href}`);\r\n return result;\r\n }\r\n else {\r\n console.log(`Failed DELETE from /tenant: ${url.href}`);\r\n result.error = await processErrors(response);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(result.error);\r\n }\r\n return result;\r\n}\r\n//tenantsGet\r\nexport async function tenantsGet(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n workspaceID: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we need a workspace id\r\n if (workspaceID === \"\") {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"tenantsGet: no workspace provided\";\r\n return result;\r\n }\r\n // create endpoint\r\n let endpoint: string = mindlineConfig.tenantsEndpoint();\r\n // add parameter to endpoint\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"workspaceId\", workspaceID);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n // make endpoint call\r\n let options = { method: \"GET\", headers: headers };\r\n try {\r\n if (debug) debugger;\r\n console.log(`Attempting GET from /tenants: ${url.href}`);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let returnedArray: Array<Object> = await response.json();\r\n if (returnedArray != null) {\r\n result.array = returnedArray;\r\n let initialValue: string = \"\";\r\n console.log(`Successful GET from /tenants: ${result.array.reduce((acc, curr: any) => acc + curr.domain + \" \", initialValue)}`);\r\n return result;\r\n }\r\n else {\r\n result.error = `Failed GET from /tenants: failed to JSON-parse response`;\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n else {\r\n console.log(`Failed GET from /tenants: ${url.href}`);\r\n result.error = await processErrors(response);\r\n result.status = 500;\r\n result.result = false;\r\n console.log(result.error);\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//tenantPost: write validated tenant to back end\r\nexport async function tenantPost(\r\n instance: IPublicClientApplication,\r\n addingUser: User,\r\n tenant: Tenant,\r\n workspaceId: string\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we expect valid tid, name, validated domain\r\n if (tenant.tid === \"\" || tenant.name === \"\" || tenant.domain === \"\") {\r\n result.result = false;\r\n result.error = \"tenantPost: invalid tid, name, domain\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create parametrized tenant endpoint\r\n let endpoint: string = mindlineConfig.tenantEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"workspaceId\", workspaceId);\r\n // create tenant headers\r\n const headers = await mindlineDefineHeaders(instance, addingUser);\r\n // create tenant body\r\n let tenantBody: string = `\r\n {\"tenantId\": \"${tenant.tid}\", \r\n \"name\": \"${tenant.name}\", \r\n \"domain\": \"${tenant.domain}\",\r\n \"type\": 1,\r\n \"authority\": \"${tenant.authority}\"\r\n }`;\r\n let options = { method: \"POST\", headers: headers, body: tenantBody };\r\n // make tenant endpoint call\r\n try {\r\n console.log(`Attempting POST to ${url.href}: ${tenantBody}`);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful POST to ${url.href}: ${tenantBody}`);\r\n return result;\r\n }\r\n else {\r\n console.log(`Failed POST to ${url.href}: ${tenantBody}`);\r\n result.error = await processErrors(response);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(result.error);\r\n }\r\n return result;\r\n}\r\n//workspacePut\r\nexport async function workspacePut(instance: IPublicClientApplication, authorizedUser: User, workspaceId: string, workspaceName: string): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (workspaceId == \"\" || workspaceName == \"\") {\r\n result.result = false;\r\n result.error = \"workspacePut: invalid workspace ID or name\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create parameterized workspace endpoint\r\n let endpoint: string = mindlineConfig.workspaceEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"workspaceId\", workspaceId);\r\n url.searchParams.append(\"workspaceName\", workspaceName);\r\n // create workspace headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n let options = { method: \"PUT\", headers: headers };\r\n // make config endpoint call\r\n try {\r\n console.log(\"Attempting PUT to /workspace: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful PUT to ${url.href}`);\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed PUT to ${url.href}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//workspacesGet\r\nexport async function workspacesGet(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we need a valid email address\r\n if (user.mail == undefined || user.mail == \"\") {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"adminsGet: no workspace provided\";\r\n return result;\r\n }\r\n // create workspaces endpoint\r\n let endpoint: string = mindlineConfig.workspacesEndpoint();\r\n // create workspace endpoint\r\n let url: URL = new URL(endpoint);\r\n // create workspace headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n // make workspace endpoint call\r\n let options = { method: \"GET\", headers: headers };\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting GET from /workspaces endpoint: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let returnedArray: Array<Object> = await response.json();\r\n if (returnedArray != null) {\r\n result.array = returnedArray;\r\n let initialValue: string = \"\";\r\n console.log(`Successful GET from /workspaces: ${result.array.reduce((acc, curr: any) => acc + curr.name + \" \", initialValue)}`);\r\n return result;\r\n }\r\n else {\r\n result.error = `Failed GET from /workspaces: failed to JSON-parse response`;\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n else {\r\n console.log(`Failed GET from /workspaces: ${url.href}`);\r\n result.error = await processErrors(response);\r\n result.status = 500;\r\n result.result = false;\r\n console.log(result.error);\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//readerPost\r\nexport async function readerPost(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n config: SyncConfig\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (instance == null || authorizedUser == null) {\r\n result.result = false;\r\n result.error = \"readerPost: invalid parameters\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create reader endpoint\r\n let readerEndpoint: string = mindlineConfig.readerStartSyncEndpoint();\r\n let url: URL = new URL(readerEndpoint);\r\n url.searchParams.append(\"configurationId\", config.id);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // make reader endpoint call\r\n let options = { method: \"POST\", headers: headers };\r\n try {\r\n console.log(\"Attempting POST to /startSync: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful POST to /startSync: ${readerEndpoint}`);\r\n let jsonResponse = await response.json();\r\n if (jsonResponse.PayloadStr != \"\") {\r\n result.array = JSON.parse(jsonResponse.PayloadStr);\r\n }\r\n else {\r\n result.result = false;\r\n result.error = \"readerPost: blank payload returned, sync may be disabled on back end\";\r\n result.status = 500;\r\n }\r\n return result;\r\n } else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed POST to /startSync: ${readerEndpoint}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n } catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}"],"names":["FILTER_FIELD","sum","b","helloNpm","getSyncVersion","APIResult","__publicField","azureConfig","_mindlineConfig","mindlineConfig","graphConfig","Group","UserScope","a","User","TenantType","TenantPermissionType","Tenant","getAppId","authority","tct","getGraphEndpoint","getLoginEndpoint","TenantConfigType","TenantConfigInfo","SyncConfig","AuditConfig","Workspace","storageAvailable","storage","x","e","InitInfo","bClearLocalStorage","__privateAdd","_InitInfo_instances","result","iiReadFromLocalStorage","__privateMethod","initFromObjects_fn","usersString","users","tenantsString","tenants","configsString","configs","workspacesString","workspaces","deserializeArray","initInfoString","item","workspace","userID","user","currentUser","tenantID","tenant","currentTenant","configID","config","currentConfig","ii","newuser","newtenant","newconfig","newworkspace","tasksData","TaskArray","_TaskArray_instances","Task","taskArray","initTasksFromObjects_fn","taskType","startDate","task","findTask_fn","save_fn","endDate","status","t","tasks","newTask","st","newSubtask","taskArrayString","minuteAdjustment","elapsedSeconds","elapsedMinutes","Milestone","run","start","post","read","write","MilestoneArray","_MilestoneArray_instances","milestones","syncmilestones","milestonesString","setMilestones","currentRun","newMilestone","ms","newms","removedMilestone","removed","milestone","BatchArray","syncPortalGlobalState","tciPotentialSource","sourceTenant","sourceTenantNode","TenantNode","tciPotentialTarget","targetTenant","targetTenantNode","setSyncProgress","setConfigSyncResult","setIdleText","batchIdArray","setRefreshDeltaTrigger","setReadersTotal","setReadersCurrent","setWritersTotal","setWritersCurrent","handler","message","matchingPair","o","tenantNode","statsarray","statskeys","statsvalues","bTotalCountZero","bCurrentCountZero","j","bTotalCount","bCurrentCount","bDeferredCount","bRescheduledCount","tidRegexp","matchTID","writerNode","bReadingComplete","bWritingComplete","bWritingStarted","bNothingToSync","readerTotal","readerCurrent","writerTotal","writerCurrent","expectedPercentPerSecond","batchPair","endpoint","endpointUrl","connection","signalR","error","connectionId","instance","authorizedUser","readerPost","tid","name","batchId","total","written","deferred","ResourceArray","bInitialize","resourceArray","resourceNodesString","resources","ResourceNode","readResources","raString","type","resource","cost","ActorArray","aaFromLocalStorage","actorsString","actors","ActorNode","actor","role","updatedby","updatedon","getGraphAPIScope","graphDefineHeaders","headers","graphAPIScope","response","accounts","homeAccountId","account","groupsGet","groupSearchString","options","groupsEndpoint","data","oauth2PermissionGrantsGet","spid","oid","spurl","url","oauth2PermissionGrantsSet","loggedInUser","id","scopes","grantsurl","scopesBody","requestAdminConsent","admin","adminConsentURL","clientId","servicePrincipalGet","appid","signIn","tenantUnauthenticatedLookup","signinURL","regex","regexMatch","domain","signInIncrementally","scope","challengeURL","signOut","userEndpoint","userLogoutResponse","signoutURL","tenantRelationshipsGetByDomain","debug","bearer","tenantEndpoint","tenantRelationshipsGetById","endpoints","regexes","openidEndpoint","tenantAuthEndpoint","k","authMatches","userDelegatedScopesGet","grants","userDelegatedScopesRemove","usersGet","usersEndpoint","auditConfigAdd","ac","auditConfigPost","auditConfigRetrieve","configurationId","auditConfigGet","auditConfigEdit","auditConfigPut","auditConfigRemove","auditConfigDelete","auditEventsRetrieve","auditEventsGet","configEdit","setConfigId","setSelectedConfigs","configPost","newSelection","idx","configPut","configEnable","enabled","configPatch","configConsentForRead","configId","consent","configConsentReadPut","configConsentForWrite","configConsentWritePut","configsRefresh","workspaceId","w","workspaceConfigsGet","configSelectedId","cid","c","processReturnedConfigs","configRemove","configDelete","initGet","initPost","workspaceInfoGet","tenantAdd","tenantPost","tenantRemove","tenantDelete","userAdd","adminPost","userRemove","adminDelete","workspaceEdit","workspaceName","workspacePut","processReturnedAdmins","returnedAdmins","adminSelectedId","usIndex","u","dummyIndex","processReturnedTenants","returnedTenants","tenantSelectedId","tsIndex","returnedConfigs","csIndex","tci","tenantConfigInfo","workspacesGet","wsIndex","adminsPromise","adminsGet","tenantsPromise","tenantsGet","configsPromise","configsGet","adminsResult","tenantsResult","configsResult","getPowerBIAccessToken","accesstoken","syncversion","azureDefineHeaders","i","canListRootAssignments","listrootassignmentsEndpoint","processErrors","elevateGlobalAdminToUserAccessAdmin","elevateaccessEndpoint","getAPIScope","apiAppID","mindlineDefineHeaders","apiScope","errorString","acc","curr","workspaceID","returnedArray","adminBody","configBody","json","configConsentReadBody","configConsentWriteBody","sourceGroupId","sourceGroupName","targetGroupId","targetGroupName","initBody","addingUser","tenantBody","readerEndpoint","jsonResponse"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"sync.es.js","sources":["../src/index.ts"],"sourcesContent":["//index.ts - published interface - AAD implementations, facade to Mindline SyncConfig API\r\nimport * as signalR from \"@microsoft/signalr\"\r\nimport { AccountInfo } from \"@azure/msal-common\";\r\nimport { IPublicClientApplication, AuthenticationResult } from \"@azure/msal-browser\"\r\nimport { deserializeArray } from 'class-transformer';\r\nimport users from \"./users.json\";\r\nimport tenants from \"./tenants.json\";\r\nimport configs from \"./configs.json\";\r\nimport workspaces from \"./workspaces.json\";\r\nimport syncmilestones from './syncmilestones.json';\r\nimport resources from './resources.json';\r\nimport actors from './actors.json';\r\n\r\nconst FILTER_FIELD = \"workspaceIDs\";\r\n// called by unit tests\r\nexport function sum(a: number, b: number): number {\r\n return a + b;\r\n}\r\nexport function helloNpm(): string {\r\n return \"hello NPM\";\r\n}\r\n// main application exports\r\ndeclare const __SYNC_VERSION__: string\r\nexport function getSyncVersion(): string {\r\n // injected at build time\r\n return __SYNC_VERSION__;\r\n}\r\nexport class APIResult {\r\n result: boolean;\r\n status: number;\r\n error: string;\r\n version: string;\r\n array: Array<Object> | null;\r\n constructor() { this.result = true; this.status = 200; this.error = \"\"; this.version = getSyncVersion(); this.array = null; }\r\n}\r\nexport class azureConfig {\r\n // azure graph REST API endpoints\r\n static azureElevateAccess: string = \"https://management.azure.com/providers/Microsoft.Authorization/elevateAccess?api-version=2016-07-01\";\r\n static azureListRootAssignments: string = \"https://management.azure.com/providers/Microsoft.Authorization/roleAssignments?api-version=2022-04-01&$filter=principalId+eq+\";\r\n};\r\nexport class mindlineConfig {\r\n static environmentTag: string = \"dev\";\r\n // config API endpoints\r\n static adminEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/admin`\r\n };\r\n static adminIncompleteEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/incomplete-admin`;\r\n };\r\n static adminsEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/admins`;\r\n };\r\n static auditConfigEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/audit-configuration`;\r\n }\r\n static auditEventsEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/audit-events`;\r\n };\r\n static configConsentEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/configuration/consent`;\r\n };\r\n static configEnabledEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/configuration/status`;\r\n };\r\n static configEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/configuration`;\r\n };\r\n static configsEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/configurations`;\r\n };\r\n static initEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/configuration/init`;\r\n };\r\n static readerStartSyncEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/startSync`;\r\n };\r\n static tenantEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/tenant`;\r\n };\r\n static tenantsEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/tenants`;\r\n };\r\n static workspaceEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/workspace`;\r\n };\r\n static workspacesEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/workspaces`;\r\n };\r\n static workspaceConfigsEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-configurationapi-westus.azurewebsites.net/api/v1/workspace-configurations`;\r\n };\r\n // SignalR endpoint\r\n static signalREndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-signalrdispatcher-westus.azurewebsites.net/statsHub`;\r\n };\r\n static statsEndpoint(): string {\r\n return `https://${mindlineConfig.environmentTag}-signalrdispatcher-westus.azurewebsites.net/api/stats`;\r\n }\r\n}\r\nexport class graphConfig {\r\n // graph API predicates\r\n static graphGroupsPredicate: string = \"v1.0/groups\";\r\n static graphMailPredicate: string = \"v1.0/me/messages\";\r\n static graphMePredicate: string = \"v1.0/me\";\r\n static graphOauth2PermissionGrantsPredicate: string = \"v1.0/oauth2PermissionGrants\";\r\n static graphServicePrincipalsPredicate: string = \"v1.0/servicePrincipals\";\r\n static graphUsersPredicate: string = \"v1.0/users\";\r\n static graphTenantByDomainPredicate: string = \"beta/tenantRelationships/findTenantInformationByDomainName\";\r\n static graphTenantByIdPredicate: string = \"beta/tenantRelationships/findTenantInformationByTenantId\";\r\n // authority values are based on the well-known OIDC auth endpoints\r\n static authorityWW: string = \"https://login.microsoftonline.com/\";\r\n static authorityWWRegex: RegExp = /^(https:\\/\\/login\\.microsoftonline\\.com\\/)([\\dA-Fa-f]{8}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{12})\\/oauth2\\/authorize$/;\r\n static authorityUS: string = \"https://login.microsoftonline.us/\";\r\n static authorityUSRegex: RegExp = /^(https:\\/\\/login\\.microsoftonline\\.us\\/)([\\dA-Fa-f]{8}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{12})\\/oauth2\\/authorize$/;\r\n static authorityCN: string = \"https://login.partner.microsoftonline.cn/\";\r\n static authorityCNRegex: RegExp = /^(https:\\/\\/login\\.partner\\.microsoftonline\\.cn\\/)([\\dA-Fa-f]{8}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{4}-[\\dA-Fa-f]{12})\\/oauth2\\/authorize$/;\r\n};\r\nexport class Group {\r\n id: string = \"\";\r\n displayName: string = \"\";\r\n description: string = \"\";\r\n}\r\nexport class UserScope {\r\n group: string = \"\";\r\n value: string = \"\";\r\n consented: boolean = false;\r\n removable: boolean = false;\r\n expanded: string = \"\";\r\n static compareByValue(a: UserScope, b: UserScope): number {\r\n return a.value.localeCompare(b.value);\r\n }\r\n static compareByGroup(a: UserScope, b: UserScope): number {\r\n return a.group.localeCompare(b.group);\r\n }\r\n}\r\nexport class User {\r\n oid: string;\r\n name: string;\r\n mail: string;\r\n authority: string;\r\n tid: string;\r\n companyName: string;\r\n companyDomain: string;\r\n workspaceIDs: string;\r\n session: string; // button text\r\n spacode: string; // to get front end access token\r\n graphAccessToken: string; // front end graph access token\r\n mindlineAccessToken: string; // front end mindline access token\r\n azureAccessToken: string; // front end azure access token\r\n loginHint: string; // to help sign out without prompt\r\n scopes: string[]; // to detect if incremental consent has happened\r\n authTS: Date; // timestamp user was authenticated\r\n sel: boolean; // selection state\r\n loggingIn: boolean; // set when the user is logging in to support spinner visibility\r\n constructor() {\r\n this.oid = \"\";\r\n this.name = \"\";\r\n this.mail = \"\";\r\n this.authority = \"\";\r\n this.tid = \"\";\r\n this.companyName = \"\";\r\n this.companyDomain = \"\";\r\n this.workspaceIDs = \"\";\r\n this.session = \"Sign In\";\r\n this.spacode = \"\";\r\n this.graphAccessToken = \"\";\r\n this.mindlineAccessToken = \"\";\r\n this.azureAccessToken = \"\";\r\n this.loginHint = \"\";\r\n this.scopes = new Array();\r\n this.authTS = new Date(0);\r\n this.sel = false;\r\n this.loggingIn = false;\r\n }\r\n}\r\n\r\nexport enum TenantType {\r\n invalid = 0,\r\n aad = 1,\r\n ad = 2,\r\n googleworkspace = 3\r\n}\r\ntype TenantTypeStrings = keyof typeof TenantType;\r\nexport enum TenantPermissionType {\r\n read = 1,\r\n write = 2,\r\n notassigned = 3\r\n}\r\nexport type TenantPermissionTypeStrings = keyof typeof TenantPermissionType;\r\nexport class Tenant {\r\n tid: string;\r\n name: string;\r\n domain: string;\r\n tenantType: TenantTypeStrings;\r\n authority: string;\r\n workspaceIDs: string;\r\n sel: boolean; // selection state\r\n graphSP: string; // graph resource ID (service principal) for this tenant\r\n lookupfield: string; // are we validating by TID or by domain? \"Domain\" | \"Tenant ID\"\r\n constructor() {\r\n this.tid = \"\";\r\n this.name = \"\";\r\n this.domain = \"\";\r\n this.tenantType = \"aad\";\r\n this.authority = \"\";\r\n this.workspaceIDs = \"\";\r\n this.sel = false;\r\n this.graphSP = \"\";\r\n this.lookupfield = \"Domain\";\r\n }\r\n}\r\nfunction getAppId(authority: string, tct: TenantConfigType): string {\r\n switch (authority) {\r\n case graphConfig.authorityWW: return tct === TenantConfigType.source ? \"85d35da2-4118-4b03-aa05-605cedd7f2f8\" : \"63100afe-506e-4bb2-8ff7-d8d5ab373129\";\r\n case graphConfig.authorityUS: return tct === TenantConfigType.source ? \"b08630c7-e227-4215-9746-afc9286fb864\" : \"17aa5d5a-f09f-4cec-87a6-28596f9fa513\";\r\n case graphConfig.authorityCN: return tct === TenantConfigType.source ? \"7db7293b-add9-4a3f-8562-1a20bfe27d5e\" : \"debd015b-1154-4111-a4cb-fc220a537697\";\r\n default: debugger; return \"\";\r\n }\r\n}\r\n// https://learn.microsoft.com/en-us/graph/deployments\r\nfunction getGraphEndpoint(authority: string): string {\r\n switch (authority) {\r\n case graphConfig.authorityWW: return \"https://graph.microsoft.com/\";\r\n case graphConfig.authorityUS: return \"https://graph.microsoft.us/\";\r\n case graphConfig.authorityCN: return \"https://microsoftgraph.chinacloudapi.cn/\";\r\n default: debugger; return \"\";\r\n }\r\n}\r\n// https://learn.microsoft.com/en-us/entra/identity-platform/authentication-national-cloud\r\nfunction getLoginEndpoint(authority: string): string {\r\n switch (authority) {\r\n case graphConfig.authorityWW: return \"https://login.microsoftonline.com/\";\r\n case graphConfig.authorityUS: return \"https://login.microsoftonline.us/\";\r\n case graphConfig.authorityCN: return \"https://login.partner.microsoftonline.cn/\";\r\n default: debugger; return \"\";\r\n }\r\n}\r\nexport enum TenantConfigType {\r\n source = 1,\r\n target = 2,\r\n sourcetarget = 3\r\n}\r\nexport type TenantConfigTypeStrings = keyof typeof TenantConfigType;\r\nexport class TenantConfigInfo {\r\n tid: string;\r\n sourceGroupId: string;\r\n sourceGroupName: string;\r\n targetGroupId: string;\r\n targetGroupName: string;\r\n configurationTenantType: TenantConfigTypeStrings;\r\n deltaToken: string;\r\n usersWritten: number;\r\n configId: string;\r\n batchId: string;\r\n isReadPermissionConsented: boolean;\r\n isWritePermissionConsented: boolean;\r\n constructor() {\r\n this.tid = \"\";\r\n this.sourceGroupId = \"\";\r\n this.sourceGroupName = \"\";\r\n this.targetGroupId = \"\";\r\n this.targetGroupName = \"\";\r\n this.configurationTenantType = \"source\";\r\n this.deltaToken = \"\";\r\n this.usersWritten = 0;\r\n this.configId = \"\";\r\n this.batchId = \"\";\r\n this.isReadPermissionConsented = false;\r\n this.isWritePermissionConsented = false;\r\n }\r\n}\r\nexport class SyncConfig {\r\n id: string;\r\n workspaceId: string;\r\n name: string;\r\n description: string;\r\n tenants: TenantConfigInfo[];\r\n isEnabled: boolean;\r\n workspaceIDs: string;\r\n sel: boolean; // selection state\r\n constructor() {\r\n this.id = \"\";\r\n this.name = \"\";\r\n this.workspaceId = \"\";\r\n this.description = \"\";\r\n this.tenants = new Array();\r\n this.isEnabled = false;\r\n this.workspaceIDs = \"\";\r\n this.sel = false;\r\n }\r\n}\r\nexport class AuditConfig {\r\n id: string;\r\n workspaceId: string;\r\n name: string;\r\n description: string;\r\n isEnabled: boolean;\r\n isReadPermissionConsented: boolean;\r\n email: string;\r\n tenantId: string;\r\n sel: boolean; // selection state\r\n constructor() {\r\n this.id = \"\";\r\n this.name = \"\";\r\n this.workspaceId = \"\";\r\n this.description = \"\";\r\n this.isEnabled = false;\r\n this.isReadPermissionConsented = false;\r\n this.email = \"\";\r\n this.tenantId = \"\";\r\n this.sel = false;\r\n }\r\n}\r\n\r\nexport type ConfigType = 'sync' | 'auditlogs';\r\n\r\nexport class Workspace {\r\n id: string;\r\n name: string;\r\n ownerid: string;\r\n associatedUsers: string[];\r\n associatedTenants: string[];\r\n associatedConfigs: string[];\r\n sel: boolean; // selection state\r\n constructor() {\r\n this.id = \"\";\r\n this.name = \"\";\r\n this.ownerid = \"\";\r\n this.associatedUsers = new Array();\r\n this.associatedTenants = new Array();\r\n this.associatedConfigs = new Array();\r\n this.sel = false;\r\n }\r\n}\r\n// check for localStorage availability\r\nfunction storageAvailable() {\r\n let storage;\r\n try {\r\n storage = window.localStorage;\r\n const x = \"__storage_test__\";\r\n storage.setItem(x, x);\r\n storage.removeItem(x);\r\n return true;\r\n } catch (e) {\r\n return (\r\n e instanceof DOMException &&\r\n // everything except Firefox\r\n (e.code === 22 ||\r\n // Firefox\r\n e.code === 1014 ||\r\n // test name field too, because code might not be present\r\n // everything except Firefox\r\n e.name === \"QuotaExceededError\" ||\r\n // Firefox\r\n e.name === \"NS_ERROR_DOM_QUOTA_REACHED\") &&\r\n // acknowledge QuotaExceededError only if there's something already stored\r\n storage &&\r\n storage.length !== 0\r\n );\r\n }\r\n}\r\nexport class InitInfo {\r\n version: string = getSyncVersion();\r\n tab: number = 0;\r\n us: User[] = [];\r\n ts: Tenant[] = [];\r\n cs: SyncConfig[] = [];\r\n acs: AuditConfig[] = [];\r\n ws: Workspace[] = [];\r\n configlevelconsent_configid: string = \"\";\r\n configlevelconsent_access: TenantConfigType = TenantConfigType.sourcetarget;\r\n constructor(bClearLocalStorage: boolean = false) {\r\n this.init(bClearLocalStorage);\r\n }\r\n // get initial data from localStorage or file\r\n init(bClearLocalStorage: boolean): void {\r\n console.log(`Calling InitInfo::init(bClearLocalStorage: ${bClearLocalStorage ? \"true\" : \"false\"})`);\r\n // if we have a non-zero value stored, read it from localStorage\r\n if (storageAvailable()) {\r\n let result = localStorage.getItem(\"InitInfo\");\r\n if (result != null && typeof result === \"string\" && result !== \"\") {\r\n let initInfoString: string = result;\r\n let iiReadFromLocalStorage: InitInfo = JSON.parse(initInfoString);\r\n if (iiReadFromLocalStorage.us.length !== 0) {\r\n if (bClearLocalStorage) {\r\n localStorage.removeItem(\"InitInfo\");\r\n }\r\n else {\r\n this.#initFromObjects(iiReadFromLocalStorage);\r\n return;\r\n }\r\n }\r\n }\r\n }\r\n // if storage unavailable or we were just asked to clear, read defaults to enable usable UI\r\n this.tab = 0;\r\n this.version = getSyncVersion();\r\n this.configlevelconsent_configid = \"\";\r\n this.configlevelconsent_access = TenantConfigType.sourcetarget;\r\n var usersString = JSON.stringify(users);\r\n var tenantsString = JSON.stringify(tenants);\r\n var configsString = JSON.stringify(configs);\r\n var workspacesString = JSON.stringify(workspaces);\r\n try {\r\n this.us = deserializeArray(User, usersString);\r\n this.ts = deserializeArray(Tenant, tenantsString);\r\n this.cs = deserializeArray(SyncConfig, configsString);\r\n this.ws = deserializeArray(Workspace, workspacesString);\r\n this.tagWithWorkspaces();\r\n } catch (e) {\r\n debugger;\r\n }\r\n }\r\n save(): void {\r\n let initInfoString: string = JSON.stringify(this);\r\n localStorage.setItem(\"InitInfo\", initInfoString);\r\n }\r\n tagWithWorkspaces(): boolean {\r\n // first clear everyone's workspaceIDs\r\n this.us.map((item) => item.workspaceIDs = \"\");\r\n this.ts.map((item) => item.workspaceIDs = \"\");\r\n this.cs.map((item) => item.workspaceIDs = \"\");\r\n // for each workspace tag WorkspaceIDs of associated Users, Tenants, Configs\r\n for (let workspace of this.ws) {\r\n // find matching Users to tag with this workspace\r\n for (let userID of workspace.associatedUsers) {\r\n let user = this.us.find((currentUser) => currentUser.oid === userID);\r\n if (user !== undefined) {\r\n // we found the user\r\n user[FILTER_FIELD] += workspace.id;\r\n user[FILTER_FIELD] += \" \";\r\n } else {\r\n // we should not have InitInfo missing Workspace components\r\n debugger;\r\n }\r\n }\r\n // find matching Tenants to tag with this workspace\r\n for (let tenantID of workspace.associatedTenants) {\r\n let tenant = this.ts.find((currentTenant) => currentTenant.tid === tenantID);\r\n if (tenant !== undefined) {\r\n // we found the tenant\r\n tenant[FILTER_FIELD] += workspace.id;\r\n tenant[FILTER_FIELD] += \" \";\r\n } else {\r\n // we should not have InitInfo missing Workspace components\r\n debugger;\r\n }\r\n }\r\n // find matching Configs to tag with this workspace\r\n for (let configID of workspace.associatedConfigs) {\r\n let config = this.cs.find((currentConfig) => currentConfig.id === configID);\r\n if (config !== undefined) {\r\n // we found the config\r\n config[FILTER_FIELD] += workspace.id;\r\n config[FILTER_FIELD] += \" \";\r\n } else {\r\n // we should not have InitInfo missing Workspace components\r\n debugger;\r\n }\r\n }\r\n }\r\n return true;\r\n }\r\n #initFromObjects(ii: InitInfo): void {\r\n this.tab = ii.tab;\r\n this.version = getSyncVersion();\r\n this.configlevelconsent_configid = ii.configlevelconsent_configid;\r\n this.configlevelconsent_access = ii.configlevelconsent_access;\r\n if (typeof ii.us === \"undefined\") {\r\n this.us = new Array<User>();\r\n }\r\n else {\r\n this.us = ii.us.map((user: User) => {\r\n let newuser: User = new User();\r\n newuser.oid = user.oid;\r\n newuser.name = user.name;\r\n newuser.mail = user.mail;\r\n newuser.authority = user.authority;\r\n newuser.tid = user.tid;\r\n newuser.companyName = user.companyName;\r\n newuser.companyDomain = user.companyDomain;\r\n newuser.workspaceIDs = user.workspaceIDs;\r\n newuser.session = user.session;\r\n newuser.spacode = user.spacode;\r\n newuser.graphAccessToken = user.graphAccessToken;\r\n newuser.mindlineAccessToken = user.mindlineAccessToken;\r\n newuser.azureAccessToken = user.azureAccessToken;\r\n newuser.loginHint = user.loginHint;\r\n newuser.scopes = user.scopes;\r\n newuser.authTS = new Date(user.authTS);\r\n newuser.sel = user.sel;\r\n newuser.loggingIn = user.loggingIn;\r\n return newuser\r\n });\r\n }\r\n if (typeof ii.ts === \"undefined\") {\r\n this.ts = new Array<Tenant>();\r\n }\r\n else {\r\n this.ts = ii.ts.map((tenant: Tenant) => {\r\n let newtenant: Tenant = new Tenant();\r\n newtenant.tid = tenant.tid;\r\n newtenant.name = tenant.name;\r\n newtenant.domain = tenant.domain;\r\n newtenant.tenantType = tenant.tenantType;\r\n newtenant.authority = tenant.authority;\r\n newtenant.workspaceIDs = tenant.workspaceIDs;\r\n newtenant.sel = tenant.sel;\r\n newtenant.lookupfield = tenant.lookupfield;\r\n return newtenant;\r\n });\r\n }\r\n if (typeof ii.cs === \"undefined\") {\r\n this.cs = new Array<SyncConfig>();\r\n }\r\n else {\r\n this.cs = ii.cs.map((config: SyncConfig) => {\r\n let newconfig: SyncConfig = new SyncConfig();\r\n newconfig.id = config.id;\r\n newconfig.workspaceId = config.workspaceId;\r\n newconfig.name = config.name;\r\n newconfig.description = config.description;\r\n newconfig.tenants = config.tenants;\r\n newconfig.isEnabled = config.isEnabled;\r\n newconfig.workspaceIDs = config.workspaceIDs;\r\n newconfig.sel = config.sel;\r\n return newconfig;\r\n });\r\n }\r\n if (typeof ii.ws === \"undefined\") {\r\n this.ws = new Array<Workspace>();\r\n }\r\n else {\r\n this.ws = ii.ws.map((workspace: Workspace) => {\r\n let newworkspace: Workspace = new Workspace();\r\n newworkspace.id = workspace.id;\r\n newworkspace.name = workspace.name;\r\n newworkspace.ownerid = workspace.ownerid;\r\n newworkspace.associatedUsers = workspace.associatedUsers;\r\n newworkspace.associatedTenants = workspace.associatedTenants;\r\n newworkspace.associatedConfigs = workspace.associatedConfigs;\r\n newworkspace.sel = workspace.sel;\r\n return newworkspace;\r\n });\r\n }\r\n }\r\n}\r\nconst tasksData: any[] = [\r\n {\r\n id: 1,\r\n task: \"initialization\",\r\n start: \"1970-01-01T00:00:00\",\r\n end: \"1970-01-01T00:00:00\",\r\n expected: \"0:22\",\r\n status: \"not started\",\r\n expanded: true,\r\n subtasks: [\r\n {\r\n id: 2,\r\n task: \"authenticate user\",\r\n start: \"1970-01-01T00:00:00\",\r\n end: \"1970-01-01T00:00:00\",\r\n expected: \"0:01\",\r\n status: \"not started\"\r\n },\r\n {\r\n id: 3,\r\n task: \"reload React\",\r\n start: \"1970-01-01T00:00:00\",\r\n end: \"1970-01-01T00:00:00\",\r\n expected: \"0:07\",\r\n status: \"not started\"\r\n },\r\n {\r\n id: 4,\r\n task: \"GET tenant details\",\r\n start: \"1970-01-01T00:00:00\",\r\n end: \"1970-01-01T00:00:00\",\r\n expected: \"0:01\",\r\n status: \"not started\"\r\n },\r\n {\r\n id: 5,\r\n task: \"POST config init\",\r\n start: \"1970-01-01T00:00:00\",\r\n end: \"1970-01-01T00:00:00\",\r\n expected: \"0:10\",\r\n status: \"not started\"\r\n },\r\n {\r\n id: 6,\r\n task: \"GET workspaces\",\r\n start: \"1970-01-01T00:00:00\",\r\n end: \"1970-01-01T00:00:00\",\r\n expected: \"0:04\",\r\n status: \"not started\"\r\n }\r\n ]\r\n }\r\n];\r\nexport type TaskType = \"initialization\" |\r\n \"authenticate user\" |\r\n \"reload React\" |\r\n \"GET tenant details\" |\r\n \"POST config init\" |\r\n \"GET workspaces\";\r\nexport class TaskArray {\r\n tasks: Task[];\r\n constructor(bClearLocalStorage: boolean) {\r\n this.tasks = [new Task()];\r\n this.init(bClearLocalStorage);\r\n }\r\n // get initial data from localStorage or file\r\n init(bClearLocalStorage: boolean): void {\r\n console.log(`Calling TaskArray::init(bClearLocalStorage: ${bClearLocalStorage ? \"true\" : \"false\"})`);\r\n // first clear task array\r\n this.tasks.length = 0;\r\n // then clear localStorage if we have been asked to\r\n if (bClearLocalStorage) {\r\n if (storageAvailable()) localStorage.removeItem(\"Tasks\");\r\n }\r\n // then try localStorage\r\n if (storageAvailable()) {\r\n let result = localStorage.getItem(\"Tasks\");\r\n if (result != null && typeof result === \"string\" && result !== \"\") {\r\n // properly create Tasks and Dates from retrieved string\r\n let tasksString: string = result;\r\n let taskArray: TaskArray = JSON.parse(tasksString);\r\n this.tasks = this.#initTasksFromObjects(taskArray.tasks);\r\n let l = this.tasks.length;\r\n if (l !== 0) return;\r\n }\r\n }\r\n // if here, there was nothing in localStorage, use initialization file\r\n this.tasks = this.#initTasksFromObjects(tasksData);\r\n }\r\n // set start time for a task\r\n setTaskStart(taskType: TaskType, startDate: Date): void {\r\n let task: Task | undefined = this.#findTask(taskType);\r\n if (task != undefined && task != null) {\r\n task.setStart(startDate);\r\n task.status = \"in progress\";\r\n this.#save();\r\n }\r\n else {\r\n debugger;\r\n }\r\n }\r\n // set end time for a task\r\n setTaskEnd(taskType: TaskType, endDate: Date, status: string): void {\r\n let task: Task | undefined = this.#findTask(taskType);\r\n if (task != undefined && task != null) {\r\n task.setEnd(endDate);\r\n task.status = status;\r\n this.#save();\r\n }\r\n else {\r\n debugger;\r\n }\r\n }\r\n //\r\n // private\r\n //\r\n #findTask(taskType: TaskType): Task | undefined {\r\n let task: Task | undefined = this.tasks.find(t => t.task == taskType);\r\n if (task == undefined || task == null) {\r\n for (task of this.tasks) {\r\n if (task.subtasks != undefined && task.subtasks != null) {\r\n task = task.subtasks.find(t => t.task == taskType);\r\n if (task != undefined && task != null) break;\r\n }\r\n }\r\n }\r\n return task;\r\n }\r\n #initTasksFromObjects(tasks: Task[]): Task[] {\r\n return tasks.map((t: Task) => {\r\n let newTask: Task = new Task();\r\n newTask.id = t.id;\r\n newTask.task = t.task;\r\n newTask.setStart(new Date(t.start));\r\n newTask.setEnd(new Date(t.end));\r\n newTask.expected = t.expected;\r\n newTask.status = t.status;\r\n newTask.expanded = t.expanded;\r\n if (typeof t.subtasks !== \"undefined\" && t.subtasks != null) {\r\n newTask.subtasks = t.subtasks.map((st: Task) => {\r\n let newSubtask: Task = new Task();\r\n newSubtask.id = st.id;\r\n newSubtask.task = st.task;\r\n newSubtask.setStart(new Date(st.start))\r\n newSubtask.setEnd(new Date(st.end));\r\n newSubtask.expected = st.expected;\r\n newSubtask.status = st.status;\r\n newSubtask.expanded = st.expanded;\r\n return newSubtask;\r\n })\r\n }\r\n return newTask;\r\n });\r\n }\r\n #save(): void {\r\n let taskArrayString: string = JSON.stringify(this);\r\n if (storageAvailable()) {\r\n localStorage.setItem(\"Tasks\", taskArrayString);\r\n }\r\n }\r\n}\r\nexport class Task {\r\n id: number = 0;\r\n task: string = \"\";\r\n start: Date = new Date();\r\n startDisplay: string = \"\";\r\n end: Date = new Date();\r\n endDisplay: string = \"\";\r\n elapsedDisplay: string = \"\";\r\n expected: number = 0;\r\n status: string = \"\";\r\n expanded: boolean = false;\r\n subtasks: Task[] = [];\r\n setEnd(endDate: Date): void {\r\n this.end = endDate;\r\n this.endDisplay = `${this.end.getMinutes().toString().padStart(2, \"0\")}:${this.end.getSeconds().toString().padStart(2, \"0\")}`;\r\n let minuteAdjustment: number = 0;\r\n let elapsedSeconds: number = this.end.getSeconds() - this.start.getSeconds();\r\n if (elapsedSeconds < 0) { elapsedSeconds += 60; minuteAdjustment = -1; }\r\n let elapsedMinutes: number = this.end.getMinutes() - this.start.getMinutes() + minuteAdjustment;\r\n if (elapsedMinutes < 0) elapsedMinutes += 60;\r\n this.elapsedDisplay = `${elapsedMinutes.toString().padStart(2, \"0\")}:${elapsedSeconds.toString().padStart(2, \"0\")}`;\r\n };\r\n setStart(startDate: Date): void {\r\n this.start = startDate;\r\n this.startDisplay = `${this.start.getMinutes().toString().padStart(2, \"0\")}:${this.start.getSeconds().toString().padStart(2, \"0\")}`;\r\n };\r\n}\r\n// class corresponding to an execution of a SyncConfig - a *TenantNode* for each source tenant, each with a *TenantNode* array of target tenants\r\nexport class Milestone {\r\n Run: number = 0;\r\n Start: Date = new Date();\r\n startDisplay: string = \"\";\r\n POST: Date;\r\n postDisplay: string;\r\n Read: Date;\r\n readDisplay: string;\r\n Write: Date;\r\n writeDisplay: string;\r\n Duration: Date;\r\n durationDisplay: string;\r\n constructor(run: number) {\r\n this.Run = run;\r\n this.start(\"\");\r\n this.POST = new Date();\r\n this.postDisplay = \"\";\r\n this.Read = new Date();\r\n this.readDisplay = \"\";\r\n this.Write = new Date();\r\n this.writeDisplay = \"\";\r\n this.Duration = new Date();\r\n this.durationDisplay = \"\";\r\n }\r\n start(start: string): void {\r\n start == \"\" ? this.Start = new Date() : this.Start = new Date(start);\r\n this.startDisplay = `${this.Start.getMinutes().toString().padStart(2, \"0\")}:${this.Start.getSeconds().toString().padStart(2, \"0\")}`;\r\n }\r\n post(post: string): void {\r\n post == \"\" ? this.POST = new Date() : this.POST = new Date(post);\r\n this.postDisplay = `${this.POST.getMinutes().toString().padStart(2, \"0\")}:${this.POST.getSeconds().toString().padStart(2, \"0\")}`;\r\n }\r\n read(read: string): void {\r\n read == \"\" ? this.Read = new Date() : this.Read = new Date(read);\r\n this.readDisplay = `${this.Read.getMinutes().toString().padStart(2, \"0\")}:${this.Read.getSeconds().toString().padStart(2, \"0\")}`;\r\n }\r\n write(write: string): void {\r\n write == \"\" ? this.Write = new Date() : this.Write = new Date(write);\r\n this.writeDisplay = `${this.Write.getMinutes().toString().padStart(2, \"0\")}:${this.Write.getSeconds().toString().padStart(2, \"0\")}`;\r\n this.Duration = new Date(this.Write.getTime() - this.Start.getTime());\r\n this.durationDisplay = `${this.Duration.getMinutes().toString().padStart(2, \"0\")}:${this.Duration.getSeconds().toString().padStart(2, \"0\")}`;\r\n }\r\n}\r\nexport class MilestoneArray {\r\n milestones: Milestone[] = [];\r\n constructor(bClearLocalStorage: boolean) {\r\n this.init(bClearLocalStorage);\r\n }\r\n init(bClearLocalStorage: boolean): void {\r\n // read from localstorage by default\r\n if (storageAvailable()) {\r\n let result = localStorage.getItem(\"syncmilestones\");\r\n if (result != null && typeof result === \"string\" && result !== \"\") {\r\n let milestonesString: string = result;\r\n let milestones: Object[] = JSON.parse(milestonesString);\r\n if (milestones.length !== 0) {\r\n if (bClearLocalStorage) {\r\n localStorage.removeItem(\"syncmilestones\");\r\n }\r\n else {\r\n this.#initFromObjects(milestones);\r\n return;\r\n }\r\n }\r\n }\r\n }\r\n // if storage unavailable or we were just asked to clear, read from default syncmilestone file\r\n this.#initFromObjects(syncmilestones);\r\n }\r\n save(): void {\r\n let milestonesString: string = JSON.stringify(this.milestones);\r\n if (storageAvailable()) {\r\n localStorage.setItem(\"syncmilestones\", milestonesString);\r\n }\r\n }\r\n // milestone tracking during a sync\r\n start(setMilestones: (milestones: Milestone[]) => void): void {\r\n // we should always have a milestone array and a first milestone\r\n if (this.milestones == null || this.milestones.length < 1) { debugger; return; }\r\n let currentRun: number = Number(this.milestones[0].Run);\r\n // create a new milestone and prepend to front of array\r\n let newMilestone: Milestone = new Milestone(currentRun + 1);\r\n this.milestones.unshift(newMilestone);\r\n // re-define milestone array to trigger render\r\n this.milestones = this.milestones.map((ms: Milestone) => {\r\n let newms = new Milestone(ms.Run);\r\n newms.Start = ms.Start;\r\n newms.startDisplay = ms.startDisplay;\r\n newms.POST = ms.POST;\r\n newms.postDisplay = ms.postDisplay;\r\n newms.Read = ms.Read;\r\n newms.readDisplay = ms.readDisplay;\r\n newms.Write = ms.Write;\r\n newms.writeDisplay = ms.writeDisplay;\r\n newms.Duration = ms.Duration;\r\n newms.durationDisplay = ms.durationDisplay;\r\n return newms;\r\n });\r\n setMilestones(this.milestones);\r\n console.log(`Start milestone: ${this.milestones[0].Run}:${this.milestones[0].Start}`);\r\n }\r\n unstart(setMilestones: (milestones: Milestone[]) => void): void {\r\n // we should always have a milestone array and a first milestone\r\n if (this.milestones == null || this.milestones.length < 1) { debugger; return; }\r\n // remove first milestone from front of array\r\n let removedMilestone: Milestone | undefined = this.milestones.shift();\r\n // re-define milestone array to trigger render\r\n this.milestones = this.milestones.map((ms: Milestone) => {\r\n let newms = new Milestone(ms.Run);\r\n newms.Start = ms.Start;\r\n newms.startDisplay = ms.startDisplay;\r\n newms.POST = ms.POST;\r\n newms.postDisplay = ms.postDisplay;\r\n newms.Read = ms.Read;\r\n newms.readDisplay = ms.readDisplay;\r\n newms.Write = ms.Write;\r\n newms.writeDisplay = ms.writeDisplay;\r\n newms.Duration = ms.Duration;\r\n newms.durationDisplay = ms.durationDisplay;\r\n return newms;\r\n });\r\n setMilestones(this.milestones);\r\n console.log(`Unstart removed first milestone: ${removedMilestone!.Run}:${removedMilestone!.Start}`);\r\n }\r\n post(setMilestones: (milestones: Milestone[]) => void): void {\r\n // update the post value of the first milestone\r\n if (this.milestones == null || this.milestones.length < 1) { debugger; return; }\r\n this.milestones[0].post(\"\");\r\n setMilestones(this.milestones);\r\n console.log(`POST milestone: ${this.milestones[0].Run}:${this.milestones[0].POST}`);\r\n }\r\n read(setMilestones: (milestones: Milestone[]) => void): void {\r\n if (this.milestones == null || this.milestones.length < 1) { debugger; return; }\r\n this.milestones[0].read(\"\");\r\n setMilestones(this.milestones);\r\n console.log(`Read milestone: ${this.milestones[0].Run}:${this.milestones[0].Read}`);\r\n }\r\n write(setMilestones: (milestones: Milestone[]) => void): void {\r\n if (this.milestones == null || this.milestones.length < 1) { debugger; return; }\r\n this.milestones[0].write(\"\");\r\n // while we have >10 complete milestones, remove the last\r\n while (this.milestones.length > 10) {\r\n let removed: Milestone | undefined = this.milestones.pop();\r\n console.log(`Removed milestone #${removed!.Run}: ${removed!.Start}`);\r\n }\r\n // save to localstorage\r\n this.save();\r\n // re-define milestone array to trigger render\r\n this.milestones = this.milestones.map((ms: Milestone) => {\r\n let newms = new Milestone(ms.Run);\r\n newms.Start = ms.Start;\r\n newms.startDisplay = ms.startDisplay;\r\n newms.POST = ms.POST;\r\n newms.postDisplay = ms.postDisplay;\r\n newms.Read = ms.Read;\r\n newms.readDisplay = ms.readDisplay;\r\n newms.Write = ms.Write;\r\n newms.writeDisplay = ms.writeDisplay;\r\n newms.Duration = ms.Duration;\r\n newms.durationDisplay = ms.durationDisplay;\r\n return newms;\r\n });\r\n setMilestones(this.milestones);\r\n }\r\n #initFromObjects(milestones: Object[]): void {\r\n if (milestones == null) {\r\n this.milestones = new Array();\r\n }\r\n else {\r\n this.milestones = milestones.map((milestone: any) => {\r\n let ms: Milestone = new Milestone(Number(milestone.Run));\r\n ms.start(milestone.Start);\r\n ms.post(milestone.POST);\r\n ms.read(milestone.Read);\r\n ms.write(milestone.Write);\r\n return ms;\r\n });\r\n }\r\n }\r\n}\r\nexport class BatchArray {\r\n tenantNodes: TenantNode[];\r\n pb_startTS: number;\r\n pb_progress: number;\r\n pb_increment: number;\r\n pb_idle: number;\r\n pb_idleMax: number;\r\n pb_total: number;\r\n pb_timer: NodeJS.Timeout | null;\r\n milestoneArray: MilestoneArray;\r\n constructor(\r\n config: SyncConfig | null,\r\n syncPortalGlobalState: InitInfo | null,\r\n bClearLocalStorage: boolean\r\n ) {\r\n this.tenantNodes = new Array<TenantNode>();\r\n this.init(config, syncPortalGlobalState, bClearLocalStorage);\r\n this.pb_startTS = 0;\r\n this.pb_progress = 0;\r\n this.pb_increment = 0;\r\n this.pb_timer = null;\r\n this.pb_idle = 0;\r\n this.pb_idleMax = 0;\r\n this.pb_total = 0;\r\n this.milestoneArray = new MilestoneArray(false);\r\n }\r\n // populate tenantNodes based on config tenants\r\n init(\r\n config: SyncConfig | null | undefined,\r\n syncPortalGlobalState: InitInfo | null,\r\n bClearLocalStorage: boolean\r\n ): void {\r\n console.log(\r\n `Calling BatchArray::init(config: \"${config ? config.name : \"null\"}\", bClearLocalStorage: ${bClearLocalStorage ? \"true\" : \"false\"})`\r\n );\r\n // clear localStorage if we have been asked to\r\n if (bClearLocalStorage) {\r\n if (storageAvailable()) {\r\n localStorage.removeItem(config!.name);\r\n this.milestoneArray.init(bClearLocalStorage);\r\n }\r\n }\r\n // create BatchArray if passed SyncConfig and InitInfo\r\n if (config != null &&\r\n config.tenants != null &&\r\n syncPortalGlobalState != null) {\r\n // clear batch array only if we have been passed something with which to replace it\r\n this.tenantNodes.length = 0;\r\n // create a sourceTenantNode for each Source and SourceTarget\r\n config.tenants.map((tciPotentialSource: TenantConfigInfo) => {\r\n if (\r\n tciPotentialSource.configurationTenantType === \"source\" ||\r\n tciPotentialSource.configurationTenantType === \"sourcetarget\"\r\n ) {\r\n let sourceTenant = syncPortalGlobalState.ts.find(\r\n (t) => t.tid === tciPotentialSource.tid\r\n );\r\n if (sourceTenant != null) {\r\n let sourceTenantNode: TenantNode = new TenantNode(\r\n tciPotentialSource.tid,\r\n sourceTenant.name,\r\n tciPotentialSource.batchId\r\n );\r\n this.tenantNodes.push(sourceTenantNode);\r\n } else {\r\n console.log(\r\n `Error: no tenant found for config source tenant ${config.name}`\r\n );\r\n debugger;\r\n return;\r\n }\r\n }\r\n });\r\n // create targetTenantNodes for each non-matching Target and SourceTarget\r\n this.tenantNodes.map((sourceTenantNode: TenantNode) => {\r\n config.tenants.map((tciPotentialTarget: TenantConfigInfo) => {\r\n // is this a valid target?\r\n if (\r\n tciPotentialTarget.configurationTenantType === \"target\" ||\r\n tciPotentialTarget.configurationTenantType === \"sourcetarget\"\r\n ) {\r\n // is this a valid target that does not match this source?\r\n if (tciPotentialTarget.tid !== sourceTenantNode.tid) {\r\n let targetTenant = syncPortalGlobalState.ts.find(\r\n (t) => t.tid === tciPotentialTarget.tid\r\n );\r\n if (targetTenant != null) {\r\n let targetTenantNode: TenantNode = new TenantNode(\r\n tciPotentialTarget.tid,\r\n targetTenant.name,\r\n tciPotentialTarget.batchId\r\n );\r\n sourceTenantNode.targets.push(targetTenantNode);\r\n sourceTenantNode.expanded = true;\r\n } else {\r\n console.log(\r\n `Error: no tenant found for config target tenant ${config.name}`\r\n );\r\n debugger;\r\n return;\r\n }\r\n }\r\n }\r\n });\r\n });\r\n }\r\n }\r\n initializeProgressBar(setSyncProgress: (progress: number) => void, setConfigSyncResult: (result: string) => void, setIdleText: (idleText: string) => void, setMilestones: (milestones: Milestone[]) => void): void {\r\n this.pb_startTS = Date.now();\r\n this.pb_progress = 0;\r\n this.pb_increment = .25;\r\n this.pb_idle = 0;\r\n this.pb_idleMax = 0;\r\n this.pb_total = 0;\r\n this.pb_timer = setInterval(() => {\r\n // if signalR has finished the sync, stop the timer\r\n console.log(\"this.tenantNodes\", this.tenantNodes)\r\n let isCompletedOrNothingToSync = this.tenantNodes.map((tn: TenantNode) => tn.targets.map((ta) => ta.status === \"complete\" || tn.nothingtosync).reduce((prev, next) => prev && next)).reduce((prev, next) => prev && next);\r\n if (isCompletedOrNothingToSync) {\r\n clearInterval(this.pb_timer!);\r\n this.pb_timer = null;\r\n this.pb_progress = 100;\r\n setSyncProgress(this.pb_progress);\r\n setIdleText(`Complete. [max idle: ${this.pb_idleMax}]`);\r\n }\r\n else {\r\n // if we've gone 60 seconds without a signalR message, finish the sync\r\n this.pb_total = this.pb_total + 1;\r\n this.pb_idle = this.pb_idle + 1;\r\n this.pb_idleMax = Math.max(this.pb_idle, this.pb_idleMax);\r\n setIdleText(`${this.pb_total} seconds elapsed. Last update ${this.pb_idle} seconds ago. [max idle: ${this.pb_idleMax}/60]`);\r\n if (this.pb_idle >= 60) {\r\n if (this.milestoneArray.milestones[0].Write == null) {\r\n //this.milestoneArray.write(setMilestones); -- allow sync to cntinue\r\n setConfigSyncResult(`sync continuing, but no update for ${this.pb_idle} seconds`);\r\n }\r\n }\r\n // if we get to 100, the progress bar stops but SignalR or countdown timer completes the sync\r\n if (this.pb_progress < 100) {\r\n this.pb_progress = Math.min(100, this.pb_progress + this.pb_increment);\r\n setSyncProgress(this.pb_progress);\r\n }\r\n }\r\n }, 1000);\r\n this.milestoneArray.start(setMilestones);\r\n }\r\n uninitializeProgressBar(setSyncProgress: (progress: number) => void, setConfigSyncResult: (result: string) => void, setIdleText: (idleText: string) => void, setMilestones: (milestones: Milestone[]) => void): void {\r\n this.pb_startTS = 0;\r\n this.pb_progress = 0;\r\n setSyncProgress(this.pb_progress);\r\n setConfigSyncResult(\"sync failed to execute\");\r\n this.pb_increment = 0;\r\n clearInterval(this.pb_timer!);\r\n this.pb_timer = null;\r\n this.pb_idle = 0;\r\n this.pb_idleMax = 0;\r\n setIdleText(`No updates seen for ${this.pb_idle} seconds. [max idle: ${this.pb_idleMax}]`);\r\n this.milestoneArray.unstart(setMilestones);\r\n }\r\n initializeSignalR(\r\n config: SyncConfig | null | undefined,\r\n syncPortalGlobalState: InitInfo | null,\r\n batchIdArray: Array<Object>,\r\n setRefreshDeltaTrigger: (workspace: string) => void,\r\n setReadersTotal: (readersTotal: number) => void,\r\n setReadersExcluded: (readersExcluded: number) => void,\r\n setReadersCurrent: (readersCurrent: number) => void,\r\n setWritersTotal: (writersTotal: number) => void,\r\n setWritersExcluded: (writersExcluded: number) => void,\r\n setWritersCurrent: (writersCurrent: number) => void,\r\n setMilestones: (milestones: Milestone[]) => void,\r\n setConfigSyncResult: (result: string) => void,\r\n bClearLocalStorage: boolean,\r\n message: string\r\n ): void {\r\n bClearLocalStorage = bClearLocalStorage;\r\n // we have just completed a successful POST to startSync\r\n this.milestoneArray.post(setMilestones);\r\n setConfigSyncResult(\"started sync, waiting for updates...\");\r\n // re-initialize batch array with Configuration updated by the succcessful POST to startSync\r\n this.init(config, syncPortalGlobalState, false);\r\n // define newMessage handler that can access *this*\r\n let handler = (connection: signalR.HubConnection) => (message: string) => {\r\n console.log(message);\r\n let item = JSON.parse(message);\r\n // reset the countdown timer every time we get a message\r\n this.pb_idle = 0;\r\n // find the associated tenant for this SignalR message\r\n let matchingPair: any | undefined = batchIdArray.find((o: any) => o.BatchId == item.TargetID);\r\n if (matchingPair == null) {\r\n console.log(`Batch ${item.TargetID} not found in batchIdArray.`);\r\n debugger;\r\n return;\r\n }\r\n let tenantNode: TenantNode | undefined = this.tenantNodes.find((t: TenantNode) => t.tid === matchingPair.SourceId);\r\n if (tenantNode == null) { // null OR undefined\r\n console.log(`Tenant ${matchingPair.SourceId} not found in BatchArray.`);\r\n debugger;\r\n return;\r\n }\r\n tenantNode.batchId = matchingPair.BatchId;\r\n // process stats for this SignalR message (one batch per tenant node)\r\n let statsarray = item.Stats; // get the array of statistics\r\n let statskeys = Object.keys(statsarray); // get the keys of the array\r\n let statsvalues = Object.values(statsarray); // get the values of the array\r\n // does this tenantnode/batch have nothing to sync?\r\n let bTotalCountZero: boolean = false;\r\n let bCurrentCountZero: boolean = false;\r\n for (let j = 0; j < statskeys.length; j++) {\r\n let bTotalCount = statskeys[j].endsWith(\"TotalCount\");\r\n let bCurrentCount = statskeys[j].endsWith(\"CurrentCount\");\r\n let bExcludedCount = statskeys[j].endsWith(\"ExtCount\");\r\n let bDeferredCount = statskeys[j].endsWith(\"DeferredCount\");\r\n let bRescheduledCount = statskeys[j].endsWith(\"RescheduledCount\");\r\n if (bExcludedCount) {\r\n tenantNode.excluded = Math.max(Number(statsvalues[j]), tenantNode.excluded);\r\n tenantNode.targets.map(writerNode => {\r\n writerNode.excluded = tenantNode.excluded;\r\n writerNode.update(writerNode.total, writerNode.read, writerNode.excluded, writerNode.written, writerNode.deferred);\r\n })\r\n }\r\n if (bTotalCount) {\r\n bTotalCountZero = Number(statsvalues[j]) == 0;\r\n tenantNode.total = Math.max(Number(statsvalues[j]), tenantNode.total);\r\n tenantNode.targets.map(writerNode => {\r\n writerNode.total = tenantNode.total;\r\n writerNode.update(writerNode.total, writerNode.read, writerNode.excluded, writerNode.written, writerNode.deferred);\r\n })\r\n }\r\n if (statskeys[j].startsWith(\"Reader\")) {\r\n // parse tid from Reader key\r\n let tidRegexp = /Reader\\/TID:(.+)\\/TotalCount/;\r\n if (bCurrentCount) tidRegexp = /Reader\\/TID:(.+)\\/CurrentCount/;\r\n if (bExcludedCount) tidRegexp = /Reader\\/TID:(.+)\\/ExtCount/;\r\n if (bDeferredCount) tidRegexp = /Reader\\/TID:(.+)\\/DeferredCount/;\r\n if (bRescheduledCount) tidRegexp = /Reader\\/TID:(.+)\\/RescheduledCount/;\r\n let matchTID = statskeys[j].match(tidRegexp);\r\n if (matchTID == null) {\r\n console.log(`tid not found in ${statskeys[j]}.`);\r\n debugger;\r\n return;\r\n }\r\n if (bTotalCount) {\r\n bTotalCountZero = Number(statsvalues[j]) == 0;\r\n tenantNode.total = Math.max(Number(statsvalues[j]), tenantNode.total);\r\n console.log(`----- ${tenantNode.name} TID: ${tenantNode.tid} batchId: ${tenantNode.batchId}`);\r\n console.log(`----- ${tenantNode.name} Total To Read: ${tenantNode.total}`);\r\n }\r\n else {\r\n bCurrentCountZero = Number(statsvalues[j]) == 0;\r\n if (bCurrentCount) {\r\n tenantNode.read = Math.max(Number(statsvalues[j]), tenantNode.read);\r\n console.log(`----- ${tenantNode.name} Currently Read: ${tenantNode.read}`);\r\n } else if (bDeferredCount) {\r\n tenantNode.deferred = Math.max(Number(statsvalues[j]), tenantNode.deferred);\r\n console.log(`----- ${tenantNode.name} Deferred: ${tenantNode.deferred}`);\r\n }\r\n }\r\n }\r\n tenantNode.nothingtosync = bTotalCountZero && bCurrentCountZero;\r\n if (statskeys[j].startsWith(\"Writer\")) {\r\n // parse tid from Writer key\r\n let tidRegexp = /Reader\\/TID:(.+)\\/TotalCount/;\r\n if (bCurrentCount) tidRegexp = /Writer\\/TID:(.+)\\/CurrentCount/;\r\n if (bExcludedCount) tidRegexp = /Writer\\/TID:(.+)\\/ExtCount/;\r\n if (bDeferredCount) tidRegexp = /Writer\\/TID:(.+)\\/DeferredCount/;\r\n if (bRescheduledCount) tidRegexp = /Writer\\/TID:(.+)\\/RescheduledCount/;\r\n let matchTID: RegExpMatchArray | null = statskeys[j].match(tidRegexp);\r\n if (matchTID == null) {\r\n console.log(`tid not found in ${statskeys[j]}.`);\r\n debugger;\r\n return;\r\n }\r\n // this Writer node should exist precisely under the Reader for this SignalR message\r\n let writerNode: TenantNode | undefined = tenantNode.targets.find((t: TenantNode) => t.tid === matchTID[1]);\r\n if (writerNode == null) {\r\n console.log(`Writer ${tenantNode.name} not found under Reader ${tenantNode.name}.`);\r\n debugger;\r\n return;\r\n }\r\n writerNode.total = Math.max(Number(tenantNode.total), writerNode.total);\r\n writerNode.total = Math.max(Number(tenantNode.total), writerNode.total);\r\n writerNode.batchId = matchingPair.BatchId;\r\n if (bTotalCount) {\r\n writerNode.total = Math.max(Number(bTotalCount), writerNode.total);\r\n console.log(`----- ${writerNode.name} TID: ${writerNode.tid} batchId: ${writerNode.batchId}`);\r\n console.log(`----- ${writerNode.name} Total To Write: ${writerNode.total}`);\r\n }\r\n else if (bCurrentCount) {\r\n writerNode.written = Math.max(Number(statsvalues[j]), writerNode.written);\r\n console.log(`----- ${writerNode.name} Total Written: ${writerNode.written}`);\r\n }\r\n else if (bDeferredCount || bRescheduledCount) {\r\n writerNode.deferred = Math.max(Number(statsvalues[j]), writerNode.deferred);\r\n console.log(`----- ${writerNode.name} Total Deferred: ${writerNode.deferred}`);\r\n }\r\n else {\r\n console.log(`unknown writer type`);\r\n debugger;\r\n return;\r\n }\r\n writerNode.update(writerNode.total, writerNode.read, writerNode.excluded, writerNode.written, writerNode.deferred);\r\n }\r\n }\r\n // update status based on all updates in this message\r\n tenantNode.update(tenantNode.total, tenantNode.read, tenantNode.excluded, tenantNode.written, tenantNode.deferred);\r\n\r\n // for each message, enumerate nodes to assess completion state\r\n let bReadingComplete: boolean = true;\r\n let bWritingComplete: boolean = true;\r\n let bWritingStarted: boolean = false;\r\n let bNothingToSync: boolean = true;\r\n let readerTotal: number = 0;\r\n let readerCurrent: number = 0;\r\n let readerExcluded: number = 0;\r\n let writerTotal: number = 0;\r\n let writerCurrent: number = 0;\r\n let writerExcluded: number = 0;\r\n this.tenantNodes.map((sourceTenantNode: TenantNode) => {\r\n sourceTenantNode.targets.map((writerNode: TenantNode) => {\r\n bWritingComplete &&= (writerNode.status == \"complete\" || writerNode.status == \"failed\");\r\n bWritingStarted ||= (writerNode.total > 0 || writerNode.status != \"not started\");\r\n writerTotal += Math.max(writerNode.total, sourceTenantNode.total);\r\n writerCurrent += writerNode.written;\r\n writerExcluded += writerNode.excluded;\r\n });\r\n bNothingToSync &&= sourceTenantNode.nothingtosync;\r\n bReadingComplete &&= (sourceTenantNode.status == \"complete\" || sourceTenantNode.status == \"failed\");\r\n readerTotal += sourceTenantNode.total;\r\n readerCurrent += sourceTenantNode.read;\r\n readerExcluded += sourceTenantNode.excluded;\r\n });\r\n // set linear gauge max and current values\r\n setReadersTotal(readerTotal);\r\n setReadersCurrent(readerCurrent);\r\n setReadersExcluded(readerExcluded);\r\n setWritersTotal(Math.max(writerTotal, readerTotal));\r\n setWritersCurrent(writerCurrent);\r\n setWritersExcluded(writerExcluded);\r\n // check to see if there was nothing to sync\r\n if (bNothingToSync) {\r\n this.milestoneArray.write(setMilestones);\r\n connection.stop();\r\n setConfigSyncResult(\"nothing to sync\");\r\n console.log(`Setting config sync result: \"nothing to sync\"`);\r\n }\r\n else {\r\n // because it is an important milestone, we always check if we have *just* completed reading\r\n if (bReadingComplete) {\r\n this.milestoneArray.read(setMilestones);\r\n setConfigSyncResult(\"reading complete\");\r\n console.log(`Setting config sync result: \"reading complete\"`);\r\n // trigger refresh delta tokens\r\n setRefreshDeltaTrigger(config!.workspaceId);\r\n // change to % per second to complete in 12x as long as it took to get here\r\n let readTS = Date.now();\r\n let secsElapsed = (readTS - this.pb_startTS) / 1000;\r\n let expectedPercentDone = 8.5;\r\n let expectedPercentPerSecond = secsElapsed / expectedPercentDone;\r\n this.pb_increment = expectedPercentPerSecond;\r\n console.log(`Setting increment: ${this.pb_increment}% per second`);\r\n }\r\n // with that out of the way, is writing complete?\r\n if (bWritingComplete) {\r\n this.milestoneArray.write(setMilestones);\r\n connection.stop();\r\n setConfigSyncResult(\"sync complete\");\r\n console.log(`Setting config sync result: \"complete\"`);\r\n }\r\n // if not, has writing even started?\r\n else if (bWritingStarted) {\r\n setConfigSyncResult(\"writing in progress\");\r\n console.log(`Setting config sync result: \"writing in progress\"`);\r\n }\r\n // else, we must be reading (unless we already completed reading)\r\n else if (this.milestoneArray.milestones[0].Read == null) {\r\n setConfigSyncResult(\"reading in progress\");\r\n console.log(`Setting config sync result: \"reading in progress\"`);\r\n }\r\n }\r\n }\r\n\r\n // start SignalR connection based on each batchId\r\n batchIdArray.map((batchPair: any) => {\r\n const endpoint: string = mindlineConfig.signalREndpoint();\r\n let endpointUrl: URL = new URL(endpoint);\r\n endpointUrl.searchParams.append(\"statsId\", batchPair.BatchId);\r\n console.log(`Creating SignalR Hub for TID: ${batchPair.SourceId} ${endpointUrl.href}`);\r\n const connection: signalR.HubConnection = new signalR.HubConnectionBuilder()\r\n .withUrl(endpointUrl.href)\r\n .withAutomaticReconnect()\r\n .configureLogging(signalR.LogLevel.Information)\r\n .build();\r\n // when you get a message, process the message\r\n if (!!message && JSON.parse(message).TargetID === batchPair.BatchId) {\r\n handler(connection)(message)\r\n }\r\n connection.on(\"newMessage\", handler(connection));\r\n connection.onreconnecting(error => {\r\n console.assert(connection.state === signalR.HubConnectionState.Reconnecting);\r\n console.log(`Connection lost due to error \"${error}\". Reconnecting.`);\r\n });\r\n connection.onreconnected(connectionId => {\r\n console.assert(connection.state === signalR.HubConnectionState.Connected);\r\n console.log(`Connection reestablished. Connected with connectionId \"${connectionId}\".`);\r\n });\r\n // restart when you get a close event\r\n connection.onclose(async () => {\r\n console.log(`Connection closing. Attempting restart.`);\r\n await connection.start();\r\n });\r\n // start and display any caught exceptions in the console\r\n connection.start().catch(console.error);\r\n });\r\n }\r\n // start a sync cycle\r\n async startSync(instance: IPublicClientApplication, authorizedUser: User | null | undefined, config: SyncConfig | null | undefined): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (this.tenantNodes == null || this.tenantNodes.length == 0) {\r\n // we should not have an empty batch array for a test\r\n debugger;\r\n result.result = false;\r\n result.error = \"startSync: invalid parameters\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // execute post to reader endpoint\r\n result = await readerPost(instance, authorizedUser!, config!);\r\n return result;\r\n }\r\n}\r\nexport class TenantNode {\r\n expanded: boolean;\r\n status: string = \"\";\r\n name: string;\r\n tid: string;\r\n batchId: string;\r\n total: number = 0;\r\n read: number = 0;\r\n excluded: number = 0;\r\n written: number = 0;\r\n deferred: number = 0;\r\n nothingtosync: boolean;\r\n targets: TenantNode[];\r\n constructor(tid: string, name: string, batchId: string) {\r\n this.expanded = false;\r\n this.name = name;\r\n this.tid = tid;\r\n this.batchId = batchId;\r\n this.nothingtosync = false;\r\n this.targets = new Array<TenantNode>();\r\n this.update(0, 0, 0, 0, 0);\r\n }\r\n update(total: number, read: number, excluded: number, written: number, deferred: number): void {\r\n this.total = total;\r\n this.read = read;\r\n this.excluded = excluded;\r\n this.written = written;\r\n this.deferred = deferred;\r\n if (this.read === 0 && this.written === 0) this.status = \"not started\";\r\n if (this.read > 0) {\r\n if (this.read + this.excluded < this.total) this.status = \"in progress\";\r\n else if (this.read + this.excluded === this.total) this.status = \"complete\";\r\n }\r\n else if (this.written > 0) {\r\n if (this.written + this.deferred + this.excluded < this.total) this.status = \"in progress\";\r\n else if (this.written + this.deferred + this.excluded === this.total) this.status = \"complete\";\r\n else if (this.written + this.deferred + this.excluded >= this.total) this.status = \"failed\";\r\n }\r\n }\r\n}\r\nexport class ResourceArray {\r\n resourceNodes: ResourceNode[];\r\n constructor(bInitialize: boolean, bClearLocalStorage: boolean) {\r\n this.resourceNodes = new Array<ResourceNode>();\r\n if (bInitialize) {\r\n this.init(bClearLocalStorage);\r\n }\r\n }\r\n // get resource data from localStorage or file\r\n init(bClearLocalStorage: boolean): void {\r\n console.log(`Calling ResourceArray::init(bClearLocalStorage: ${bClearLocalStorage ? \"true\" : \"false\"})`);\r\n // if we have a non-empty string value stored, read it from localStorage\r\n if (storageAvailable()) {\r\n let result = localStorage.getItem(\"ResourceArray\");\r\n if (result != null && typeof result === \"string\" && result !== \"\") {\r\n if (bClearLocalStorage) {\r\n localStorage.removeItem(\"ResourceArray\");\r\n }\r\n else {\r\n // read entire object from localstorage\r\n let raString: string = result;\r\n let resourceArray: ResourceArray = JSON.parse(raString);\r\n this.resourceNodes = resourceArray.resourceNodes;\r\n return;\r\n }\r\n }\r\n }\r\n // if storage unavailable or we were just asked to clear, read resources from file\r\n var resourceNodesString = JSON.stringify(resources);\r\n try {\r\n this.resourceNodes = deserializeArray(ResourceNode, resourceNodesString);\r\n } catch (e) {\r\n debugger;\r\n }\r\n }\r\n // read\r\n async read(instance: IPublicClientApplication, user: User): Promise<ResourceArray> {\r\n let resources: ResourceArray = new ResourceArray(false, false);\r\n resources.resourceNodes = await readResources(instance, user);\r\n return resources;\r\n }\r\n // save resource data to localstorage\r\n save(): void {\r\n // if we have localStorage, save resources\r\n if (storageAvailable()) {\r\n let raString: string = JSON.stringify(this);\r\n localStorage.setItem(\"ResourceArray\", raString);\r\n }\r\n }\r\n}\r\nexport class ResourceNode {\r\n type: string;\r\n resource: string;\r\n cost: number;\r\n expanded: boolean;\r\n resources: ResourceNode[];\r\n constructor(type: string, resource: string, cost: number) {\r\n this.type = type;\r\n this.resource = resource;\r\n this.cost = cost;\r\n this.expanded = false;\r\n this.resources = new Array<ResourceNode>();\r\n }\r\n}\r\nexport class ActorArray {\r\n actorNodes: ActorNode[];\r\n constructor(bClearLocalStorage: boolean) {\r\n this.actorNodes = new Array<ActorNode>();\r\n this.init(bClearLocalStorage);\r\n }\r\n // get initial data from localStorage or file\r\n init(bClearLocalStorage: boolean): void {\r\n console.log(`Calling ResourceArray::init(bClearLocalStorage: ${bClearLocalStorage ? \"true\" : \"false\"})`);\r\n // if we have a non-empty string value stored, read it from localStorage\r\n if (storageAvailable()) {\r\n let result = localStorage.getItem(\"RBACActors\");\r\n if (result != null && typeof result === \"string\" && result !== \"\") {\r\n if (bClearLocalStorage) {\r\n localStorage.removeItem(\"RBACActors\");\r\n }\r\n else {\r\n let actorArrayString: string = result;\r\n let aaFromLocalStorage: ActorArray = JSON.parse(actorArrayString);\r\n this.actorNodes = aaFromLocalStorage.actorNodes;\r\n return;\r\n }\r\n }\r\n }\r\n // if storage unavailable or we were just asked to clear, read defaults to enable usable UI\r\n var actorsString = JSON.stringify(actors);\r\n try {\r\n this.actorNodes = deserializeArray(ActorNode, actorsString);\r\n } catch (e) {\r\n debugger;\r\n }\r\n }\r\n}\r\nexport class ActorNode {\r\n type: string;\r\n actor: string;\r\n resource: string;\r\n role: string;\r\n updatedby: string;\r\n updatedon: string;\r\n actors: ActorNode[];\r\n constructor(type: string, actor: string, resource: string, role: string, updatedby: string, updatedon: string) {\r\n this.type = type;\r\n this.actor = actor;\r\n this.resource = resource;\r\n this.role = role;\r\n this.updatedby = updatedby;\r\n this.updatedon = updatedon;\r\n this.actors = new Array<ActorNode>();\r\n }\r\n}\r\n// ======================= Azure AD Graph API ===============================\r\n// helper functions\r\nfunction getGraphAPIScope(user: User): string {\r\n user = user;\r\n return \"Group.Read.All User.Read.All openid profile offline_access User.Read Contacts.Read CrossTenantInformation.ReadBasic.All\";\r\n}\r\n// TODO: this is where you want to trigger a re-authentication if token expires\r\nasync function graphDefineHeaders(\r\n instance: IPublicClientApplication,\r\n user: User\r\n): Promise<Headers> {\r\n const headers = new Headers();\r\n headers.append(\"Content-Type\", \"application/json\");\r\n headers.append(\"accept\", \"*/*\");\r\n const graphAPIScope: string = getGraphAPIScope(user);\r\n // only call acquireTokenByCode if we have never redeemed the code\r\n if (user.graphAccessToken == null || user.graphAccessToken === \"\") {\r\n try {\r\n let response: AuthenticationResult = await instance.acquireTokenByCode({\r\n code: user.spacode,\r\n });\r\n user.graphAccessToken = response.accessToken; // cache access token\r\n console.log(\"Front end token acquired by code: \" + user.graphAccessToken.slice(0, 20));\r\n }\r\n catch (error: any) {\r\n console.log(\"Front end token failure: \" + error);\r\n }\r\n }\r\n // otherwise, call acquireTokenSilent and deal with token expiration on exception\r\n else {\r\n try {\r\n let accounts: AccountInfo[] = instance.getAllAccounts();\r\n let homeAccountId = user.oid + \".\" + user.tid;\r\n let account: AccountInfo | undefined | null = null;\r\n for (let i: number = 0; i < accounts.length; i++) {\r\n if (accounts[i].homeAccountId == homeAccountId) {\r\n account = accounts[i];\r\n }\r\n }\r\n let response: AuthenticationResult = await instance.acquireTokenSilent({\r\n scopes: [graphAPIScope],\r\n account: account!\r\n });\r\n user.graphAccessToken = response.accessToken; // cache access token\r\n console.log(\"Front end token graph acquired silently: \" + user.graphAccessToken.slice(0, 20));\r\n }\r\n catch (error: any) {\r\n try {\r\n console.log(\"Front end graph token silent acquisition failure: \" + error);\r\n // fallback to redirect if silent acquisition fails\r\n let accounts: AccountInfo[] = instance.getAllAccounts();\r\n let homeAccountId = user.oid + \".\" + user.tid;\r\n let account: AccountInfo | null = null;\r\n for (let i: number = 0; i < accounts.length; i++) {\r\n if (accounts[i].homeAccountId == homeAccountId) {\r\n account = accounts[i];\r\n }\r\n }\r\n // assumption: this redirect will trigger login flow callbacks in program.cs\r\n instance.acquireTokenRedirect({\r\n scopes: [graphAPIScope],\r\n account: account!\r\n });\r\n }\r\n catch (error: any) {\r\n console.log(\"Front end graph token redirect acquisition failure: \" + error);\r\n }\r\n }\r\n }\r\n headers.append(\"Authorization\", `Bearer ${user.graphAccessToken}`);\r\n return headers;\r\n}\r\nexport async function groupsGet(instance: IPublicClientApplication, user: User | undefined, groupSearchString: string): Promise<{ groups: Group[], error: string }> {\r\n // need a logged in user to get graph users\r\n if (user == null || user.spacode == \"\") {\r\n return { groups: [], error: `500: invalid user passed to groupsGet` };\r\n }\r\n // create headers\r\n const headers = await graphDefineHeaders(instance, user);\r\n let options = { method: \"GET\", headers: headers };\r\n // make /groups endpoint call\r\n try {\r\n let groupsEndpoint: string = getGraphEndpoint(user.authority) + graphConfig.graphGroupsPredicate;\r\n groupsEndpoint += `/?$filter=startsWith(displayName, '${groupSearchString}')`;\r\n let response = await fetch(groupsEndpoint, options);\r\n let data = await response.json();\r\n if (typeof data.error !== \"undefined\") {\r\n return { groups: [], error: `${data.error.code}: ${data.error.message}` };\r\n }\r\n return { groups: data.value, error: `` };\r\n }\r\n catch (error: any) {\r\n console.log(error);\r\n return { groups: [], error: `Exception: ${error}` };\r\n }\r\n}\r\nexport async function oauth2PermissionGrantsGet(options: RequestInit, user: User, spid: string, oid: string): Promise<{ grants: string | null, id: string | null, error: string }> {\r\n try {\r\n // make /oauth2PermissionGrants endpoint call\r\n let spurl: string = getGraphEndpoint(user.authority) + graphConfig.graphOauth2PermissionGrantsPredicate;\r\n let url: URL = new URL(spurl);\r\n url.searchParams.append(\"$filter\", `resourceId eq '${spid}' and consentType eq 'Principal' and principalId eq '${oid}'`);\r\n let response = await fetch(url.href, options);\r\n let data = await response.json();\r\n if (typeof data.error != \"undefined\") {\r\n return { grants: null, id: null, error: `${data.error.code}: ${data.error.message}` };\r\n }\r\n // we assume there is only one such grant\r\n if (data.value.length != 1) {\r\n debugger;\r\n return { grants: null, id: null, error: `oauth2PermissionGrantsGet: more than one matching delegated consent grant.` };\r\n }\r\n return { grants: data.value[0].scope, id: data.value[0].id, error: `` };\r\n }\r\n catch (error: any) {\r\n console.log(error);\r\n return { grants: null, id: null, error: `Exception: ${error}` };\r\n }\r\n}\r\nexport async function oauth2PermissionGrantsSet(instance: IPublicClientApplication, loggedInUser: User, id: string, scopes: string): Promise<boolean> {\r\n // need a logged in user to get graph users\r\n if (loggedInUser == null || loggedInUser.spacode == \"\") {\r\n return false;\r\n }\r\n // make /oauth2PermissionGrants endpoint call\r\n try {\r\n let grantsurl: string = getGraphEndpoint(loggedInUser.authority);\r\n grantsurl += graphConfig.graphOauth2PermissionGrantsPredicate + `/${id}`;\r\n let scopesBody: string = `{ \"scope\": \"${scopes}\" }`;\r\n const headers = await graphDefineHeaders(instance, loggedInUser);\r\n let options: RequestInit = { method: \"PATCH\", headers: headers, body: scopesBody };\r\n let response = await fetch(grantsurl, options);\r\n let data = await response.json();\r\n if (response.status == 204 && response.statusText == \"No Content\") {\r\n return true;\r\n }\r\n else {\r\n debugger;\r\n console.log(`oauth2PermissionGrantsSet: PATCH failed ${data.error.code}: ${data.error.message}`);\r\n return false;\r\n }\r\n }\r\n catch (error: any) {\r\n debugger;\r\n console.log(error);\r\n return false;\r\n }\r\n}\r\nexport function requestAdminConsent(admin: User, tct: TenantConfigType): void {\r\n //\r\n // for app permissions (app roles) we must use the /.default scope for admin consent\r\n // https://learn.microsoft.com/EN-US/azure/active-directory/develop/v2-admin-consent#:~:text=In%20order%20to%20request%20app%20permissions%2C%20you%20must%20use%20the%20/.default%20value.\r\n // https://learn.microsoft.com/en-us/answers/questions/431784/how-to-grant-application-permissions-with-dynamic\r\n //\r\n // this means that, if we want to be granular about SyncReader vs. SyncWriter permissions, we must have separate Applications\r\n // in addition, if there are permissions like Group.Read.All, that also must be admin consented\r\n // for now, this function will perform admin consent for the Reader and Writer applications, we will come back to Group.Read.All and User.Read.All\r\n //\r\n // FYI - using the Challenge endpoint for app permissions caused the call to quietly fail without error\r\n // FYI - using the Challenge endpoint for this scope also caused the call to quietly fail withour error\r\n // scope = \"63100afe-506e-4bb2-8ff7-d8d5ab373129/.default\";\r\n //\r\n // thereforce, we are assuming that, for app permissions (app roles) or delegated permissions requiring admin consent, the Microsoft Identity Web Challenge endpoint does not work and we need to perform our own redirect to the admin consent endpoint\r\n // https://learn.microsoft.com/EN-US/azure/active-directory/develop/scopes-oidc#client-credentials-grant-flow-and-default\r\n // https://learn.microsoft.com/en-us/azure/active-directory/develop/v2-oauth2-client-creds-grant-flow#request-the-permissions-from-a-directory-admin\r\n //\r\n let adminConsentURL: string = getLoginEndpoint(admin.authority);\r\n adminConsentURL += admin.tid;\r\n adminConsentURL += \"/adminconsent\";\r\n let url: URL = new URL(adminConsentURL);\r\n let clientId: string = getAppId(admin.authority, tct);\r\n url.searchParams.append(\"client_id\", clientId);\r\n url.searchParams.append(\"redirect_uri\", window.location.origin);\r\n url.searchParams.append(\"domain_hint\", admin.companyDomain);\r\n url.searchParams.append(\"login_hint\", admin.mail);\r\n window.location.assign(url.href);\r\n}\r\nexport async function servicePrincipalGet(options: RequestInit, user: User, appid: string): Promise<{ spid: string, error: string }> {\r\n try {\r\n // make /servicePrincipals endpoint call to get the Service Principal ID\r\n let spurl: string = getGraphEndpoint(user.authority);\r\n spurl += graphConfig.graphServicePrincipalsPredicate;\r\n spurl += `(appId='${appid}')`;\r\n let url: URL = new URL(spurl);\r\n url.searchParams.append(\"$select\", \"id,appId,displayName\");\r\n let response = await fetch(url.href, options);\r\n let data = await response.json();\r\n if (typeof data.error !== \"undefined\") {\r\n return { spid: \"\", error: `${data.error.code}: ${data.error.message}` };\r\n }\r\n else {\r\n return { spid: data.id, error: `` };\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(error);\r\n return { spid: \"\", error: `Exception: ${error}` };\r\n }\r\n}\r\nexport async function signIn(user: User, tasks: TaskArray): Promise<boolean> {\r\n // admin authority is blank at signIn, lookup authority real-time\r\n if (user.authority == \"\") {\r\n // lookup authority for this user (the lookup call does it based on domain, but TID works as well to find authority)\r\n let tenant: Tenant = new Tenant();\r\n tenant.domain = user.tid;\r\n let bResult: boolean = await tenantUnauthenticatedLookup(tenant, false);\r\n if (bResult) {\r\n // success, we now know cloud instance where this tenant is provisioned\r\n user.authority = tenant.authority;\r\n }\r\n else {\r\n // can't sign in without an authority\r\n debugger;\r\n return false;\r\n }\r\n }\r\n // SignIn by an admin consents the app, Challenge adds incremental permissions dynamically, but requires a consented app - TODO: TEST FRESH CONSENT EXPERIENCE\r\n let signinURL: string = window.location.href;\r\n switch (user.authority) {\r\n case graphConfig.authorityWW:\r\n signinURL += \"MicrosoftIdentity/Account/SignIn\";\r\n break;\r\n case graphConfig.authorityUS:\r\n signinURL += \"USGov/SignIn\";\r\n break;\r\n case graphConfig.authorityCN:\r\n signinURL += \"China/SignIn\";\r\n break;\r\n default:\r\n // unknown authority\r\n return false;\r\n }\r\n let url: URL = new URL(signinURL);\r\n url.searchParams.append(\"redirectUri\", window.location.origin);\r\n if (user.oid !== \"1\") {\r\n url.searchParams.append(\"loginHint\", user.mail);\r\n const regex = /@([a-zA-Z0-9.-]+\\.[a-zA-Z]{2,})$/;\r\n const regexMatch = user.mail.match(regex);\r\n let domain: string = regexMatch ? regexMatch[1] : \"organizations\";\r\n url.searchParams.append(\"domainHint\", domain);\r\n }\r\n else {\r\n // \"1\" is the dummy OID set when initializing the application from JSON. No need to provide any specific hint.\r\n // This means that a user has not yet been specified by the admin. If it had been, oid would not be \"1\".\r\n url.searchParams.append(\"domainHint\", \"organizations\");\r\n }\r\n tasks.setTaskStart(\"initialization\", new Date());\r\n tasks.setTaskStart(\"authenticate user\", new Date());\r\n window.location.assign(url.href);\r\n return true;\r\n}\r\nexport function signInIncrementally(user: User, scope: string): void {\r\n debugger;\r\n if (user.oid == \"1\") return;\r\n // for dynamic delegated permissions, we can use the Microsoft Identity Web Account Controller Challenge method\r\n let challengeURL: string = window.location.href;\r\n switch (user.authority) {\r\n case graphConfig.authorityWW:\r\n challengeURL += \"MicrosoftIdentity/Account/Challenge\";\r\n break;\r\n case graphConfig.authorityUS:\r\n challengeURL += \"USGov/Challenge\";\r\n break;\r\n case graphConfig.authorityCN:\r\n challengeURL += \"China/Challenge\";\r\n break;\r\n default:\r\n // unknown authority\r\n return;\r\n }\r\n let url: URL = new URL(challengeURL);\r\n url.searchParams.append(\"redirectUri\", window.location.origin);\r\n url.searchParams.append(\"scope\", scope);\r\n url.searchParams.append(\"domainHint\", \"organizations\");\r\n url.searchParams.append(\"loginHint\", user.mail);\r\n window.location.assign(url.href);\r\n}\r\nexport async function signOut(user: User): Promise<boolean> {\r\n if (user.oid == \"1\") return false;\r\n // set logout_hint in the .NET session for streamlined logout\r\n let userEndpoint: string = window.location.href;\r\n userEndpoint += \"user\";\r\n let url = new URL(userEndpoint);\r\n url.searchParams.append(\"oid\", user.oid);\r\n url.searchParams.append(\"tid\", user.tid);\r\n url.searchParams.append(\"loginHint\", user.loginHint);\r\n url.searchParams.append(\"verb\", \"LOGOUT\");\r\n let options = { method: \"PATCH\" };\r\n let userLogoutResponse: Response = await fetch(url.href, options);\r\n if (userLogoutResponse.status == 200 && userLogoutResponse.statusText == \"OK\") {\r\n console.log(`Successfully set admin ${user.mail} logout_hint`);\r\n }\r\n else {\r\n console.log(`Failed to set admin ${user.mail} logout_hint`);\r\n return false;\r\n }\r\n // start the logout process triggering callbacks during logout\r\n // OnRedirectToIdentityProviderForSignOut - this is where we set the logout_hint for user we are trying to logout\r\n // OnSignedOutCallbackRedirect - called when the call sucessfully completes\r\n let signoutURL: string = window.location.href;\r\n switch (user.authority) {\r\n case graphConfig.authorityWW:\r\n signoutURL += \"MicrosoftIdentity/Account/SignOut\";\r\n break;\r\n case graphConfig.authorityUS:\r\n signoutURL += \"USGov/SignOut\";\r\n break;\r\n case graphConfig.authorityCN:\r\n signoutURL += \"China/SignOut\";\r\n break;\r\n default:\r\n // unknown authority\r\n return false;\r\n }\r\n url = new URL(signoutURL);\r\n url.searchParams.append(\"redirectUri\", window.location.origin);\r\n window.location.assign(url.href);\r\n return true;\r\n}\r\n//tenantRelationshipsGetByDomain - query AAD for associated company name and id\r\nexport async function tenantRelationshipsGetByDomain(loggedInUser: User, tenant: Tenant, instance: IPublicClientApplication, debug: boolean): Promise<boolean> {\r\n if (debug) debugger;\r\n // do we already have a valid tenant name? if so, nothing to add\r\n if (tenant.name != null && tenant.name !== \"\") return false;\r\n // if needed, retrieve and cache access token\r\n if (loggedInUser.graphAccessToken != null && loggedInUser.graphAccessToken === \"\") {\r\n console.log(`tenantRelationshipsGetByDomain called with invalid logged in user: ${loggedInUser.name}`);\r\n try {\r\n let response: AuthenticationResult = await instance.acquireTokenByCode({ code: loggedInUser.spacode, scopes: [\"user.read\", \"contacts.read\", \"CrossTenantInformation.ReadBasic.All\"] });\r\n loggedInUser.graphAccessToken = response.accessToken; // cache access token on the user\r\n console.log(\"tenantRelationshipsGetByDomain: Front end token acquired: \" + loggedInUser.graphAccessToken.slice(0, 20));\r\n }\r\n catch (error: any) {\r\n console.log(\"tenantRelationshipsGetByDomain: Front end token failure: \" + error);\r\n return false; // failed to get access token, no need to re-render\r\n }\r\n }\r\n // prepare Authorization headers as part of options\r\n const headers = new Headers();\r\n const bearer = `Bearer ${loggedInUser.graphAccessToken}`;\r\n headers.append(\"Authorization\", bearer);\r\n let options = { method: \"GET\", headers: headers };\r\n // make tenant endpoint call\r\n try {\r\n // create tenant info endpoint\r\n var tenantEndpoint = getGraphEndpoint(tenant.authority) + graphConfig.graphTenantByDomainPredicate;\r\n tenantEndpoint += \"(domainName='\";\r\n tenantEndpoint += tenant.domain;\r\n tenantEndpoint += \"')\";\r\n console.log(\"tenantRelationshipsGetByDomain: Attempting GET from /findTenantInformationByDomainName:\", tenantEndpoint);\r\n let response = await fetch(tenantEndpoint, options);\r\n // status IS 200, but statusText no longer returns \"OK\" 1/26/2025\r\n if (response.status == 200) { // && response.statusText == \"OK\") {\r\n let data = await response.json();\r\n if (data) {\r\n if (data.error != null) {\r\n debugger;\r\n console.log(\"tenantRelationshipsGetByDomain: Failed GET from /findTenantInformationByDomainName: \", data.error.message);\r\n return false;\r\n }\r\n else if (data.displayName != null && data.displayName !== \"\") {\r\n // set domain information on passed tenant\r\n tenant.tid = data.tenantId;\r\n tenant.name = data.displayName;\r\n console.log(\"tenantRelationshipsGetByDomain: Successful GET from /findTenantInformationByDomainName: \", data.displayName);\r\n return true; // success, need UX to re-render\r\n }\r\n }\r\n else {\r\n console.log(\"tenantRelationshipsGetByDomain: Failed to GET from /findTenantInformationByDomainName: \", tenantEndpoint);\r\n }\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(\"Failed to GET from /findTenantInformationByDomainName: \", error);\r\n return false; // failed, no need for UX to re-render\r\n }\r\n return false; // failed, no need for UX to re-render\r\n}\r\n//tenantRelationshipsGetById - query AAD for associated company name and domain\r\nexport async function tenantRelationshipsGetById(loggedInUser: User, tenant: Tenant, ii: InitInfo, instance: IPublicClientApplication, debug: boolean): Promise<boolean> {\r\n ii = ii;\r\n console.log(\"**** tenantRelationshipsGetById\");\r\n if (debug) debugger;\r\n // if needed, retrieve and cache access token\r\n if (loggedInUser.graphAccessToken === \"\") {\r\n try {\r\n let response: AuthenticationResult = await instance.acquireTokenByCode({ code: loggedInUser.spacode, scopes: [\"user.read\", \"contacts.read\", \"CrossTenantInformation.ReadBasic.All\"] });\r\n loggedInUser.graphAccessToken = response.accessToken; // cache access token\r\n console.log(\"tenantRelationshipsGetById: Front end token acquired: \" + loggedInUser.graphAccessToken.slice(0, 20));\r\n }\r\n catch (error: any) {\r\n console.log(\"tenantRelationshipsGetById: Front end token failure: \" + error);\r\n return false; // failed to get access token, no need to re-render\r\n }\r\n }\r\n // prepare Authorization headers as part of options\r\n const headers = new Headers();\r\n const bearer = `Bearer ${loggedInUser.graphAccessToken}`;\r\n headers.append(\"Authorization\", bearer);\r\n let options = { method: \"GET\", headers: headers };\r\n // make tenant endpoint call\r\n try {\r\n // create tenant info endpoint\r\n var tenantEndpoint = getGraphEndpoint(loggedInUser.authority) + graphConfig.graphTenantByIdPredicate;\r\n tenantEndpoint += \"(tenantId='\";\r\n tenantEndpoint += tenant.tid;\r\n tenantEndpoint += \"')\";\r\n console.log(\"tenantRelationshipsGetById: Attempting GET from /findTenantInformationByTenantId:\", tenantEndpoint);\r\n let response = await fetch(tenantEndpoint, options);\r\n let data = await response.json();\r\n if (data && typeof data.displayName !== undefined && data.displayName !== \"\") {\r\n // set domain information on tenant\r\n if (tenant !== undefined) {\r\n tenant.name = data.displayName;\r\n tenant.domain = data.defaultDomainName;\r\n }\r\n else {\r\n console.log(\"tenantRelationshipsGetById: missing associated tenant for logged in user.\");\r\n debugger;\r\n }\r\n console.log(\"tenantRelationshipsGetById: Successful GET from /findTenantInformationByTenantId: \", data.displayName);\r\n return true; // success, need UX to re-render\r\n }\r\n else {\r\n console.log(\"tenantRelationshipsGetById: Failed to GET from /findTenantInformationByTenantId: \", tenantEndpoint);\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(\"tenantRelationshipsGetById: Failed to GET from /findTenantInformationByTenantId: \", error);\r\n return false; // failed, no need for UX to re-render\r\n }\r\n return false; // failed, no need for UX to re-render\r\n}\r\n//tenantUnauthenticatedLookup (from https://gettenantpartitionweb.azurewebsites.net/js/gettenantpartition.js)\r\nexport async function tenantUnauthenticatedLookup(tenant: Tenant, debug: boolean): Promise<boolean> {\r\n if (debug) debugger;\r\n // prepare the 3 endpoints and corresponding regular expressions\r\n let endpoints: string[] = [graphConfig.authorityWW, graphConfig.authorityUS, graphConfig.authorityCN];\r\n let regexes: RegExp[] = [graphConfig.authorityWWRegex, graphConfig.authorityUSRegex, graphConfig.authorityCNRegex];\r\n // make unauthenticated well-known openid endpoint call(s)\r\n let response = null;\r\n try {\r\n for (let j = 0; j < 3; j++) {\r\n // create well-known openid endpoint\r\n var openidEndpoint = endpoints[j];\r\n if (tenant.lookupfield == \"Domain\") {\r\n openidEndpoint += tenant.domain;\r\n }\r\n else if (tenant.lookupfield == \"Tenant ID\") {\r\n openidEndpoint += tenant.tid;\r\n }\r\n openidEndpoint += \"/.well-known/openid-configuration\";\r\n console.log(\"Attempting GET from openid well-known endpoint: \", openidEndpoint);\r\n response = await fetch(openidEndpoint);\r\n if (response.status == 200) {\r\n let data = await response.json();\r\n if (data) {\r\n // store tenant ID and authority\r\n var tenantAuthEndpoint = data.authorization_endpoint;\r\n // USGov/China tenants are registered in WW with USGov/China authority values!\r\n for (let k = 0; k < 3; k++) {\r\n var authMatches = tenantAuthEndpoint.match(regexes[k]);\r\n if (authMatches != null) {\r\n tenant.tid = authMatches[2];\r\n tenant.authority = authMatches[1];\r\n console.log(`Successful GET from openid well-known endpoint: tid: ${tenant.tid} authority: ${tenant.authority}`);\r\n return true; // success, need UX to re-render\r\n }\r\n }\r\n }\r\n else {\r\n console.log(`Failed JSON parse of openid well-known endpoint response ${openidEndpoint}.`);\r\n }\r\n }\r\n else {\r\n console.log(`Failed GET from ${openidEndpoint}.`);\r\n }\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(\"Failed to GET from openid well-known endpoint: \", error);\r\n }\r\n if (tenant.tid == \"\" || tenant.authority == \"\") {\r\n console.log(`GET from openid well-known endpoint failed to find tenant: ${response ? response.statusText : \"unknown\"}`);\r\n }\r\n return false; // failed, no need for UX to re-render\r\n}\r\nexport async function userDelegatedScopesGet(instance: IPublicClientApplication, loggedInUser: User, tenant: Tenant): Promise<{ scopes: string | null, id: string | null, error: string }> {\r\n // need a logged in user and valid tenant to query graph\r\n if (loggedInUser == null || loggedInUser.spacode == \"\" || tenant == null) {\r\n debugger;\r\n return { scopes: null, id: null, error: `500: invalid parameter(s) passed to getUserDelegatedScopes` };\r\n }\r\n // create headers\r\n const headers = await graphDefineHeaders(instance, loggedInUser);\r\n let options: RequestInit = { method: \"GET\", headers: headers };\r\n try {\r\n // first, cache Graph resource ID (service principal) for this tenant if we don't have it already\r\n if (tenant.graphSP == \"\") {\r\n let { spid, error } = await servicePrincipalGet(options, loggedInUser, \"00000003-0000-0000-c000-000000000000\");\r\n if (error != \"\") {\r\n debugger;\r\n return { scopes: null, id: null, error: `${error}` };\r\n }\r\n tenant.graphSP = spid;\r\n }\r\n // then, retrieve the delegated Graph permissions assigned to this user\r\n let { grants, id, error } = await oauth2PermissionGrantsGet(options, loggedInUser, tenant.graphSP, loggedInUser.oid);\r\n if (error != \"\") {\r\n debugger;\r\n return { scopes: null, id: null, error: `${error}` };\r\n }\r\n return { scopes: grants, id: id, error: `` };\r\n }\r\n catch (error: any) {\r\n debugger;\r\n console.log(error);\r\n return { scopes: null, id: null, error: `Exception: ${error}` };\r\n }\r\n}\r\nexport async function userDelegatedScopesRemove(instance: IPublicClientApplication, loggedInUser: User, tenant: Tenant, scope: string): Promise<boolean> {\r\n // need a logged in user and valid tenant to query graph\r\n if (loggedInUser == null || loggedInUser.spacode == \"\" || tenant == null) {\r\n debugger;\r\n return false;\r\n }\r\n // get current set of delegated scopes in order to remove passed scope\r\n let { scopes, id, error } = await userDelegatedScopesGet(instance, loggedInUser, tenant);\r\n if (error != \"\") {\r\n debugger;\r\n console.log(`userDelegatedScopesRemove: cannot find userDelegatedScopes for ${loggedInUser.mail}: ${error}`);\r\n return false;\r\n }\r\n // remove passed scope (case sensitive)\r\n scopes = scopes!.replace(scope, \"\");\r\n // set updated oauth2permissions\r\n let removed: boolean = await oauth2PermissionGrantsSet(instance, loggedInUser, id!, scopes);\r\n if (!removed) {\r\n debugger;\r\n console.log(`userDelegatedScopesRemove: cannot set oauth2PermissionGrants for ${loggedInUser.mail}: ${error}`);\r\n return false;\r\n }\r\n // replace scope array on logged in user\r\n loggedInUser.scopes = scopes.split(\" \");\r\n return removed;\r\n}\r\n//usersGet - GET from AAD Users endpoint\r\nexport async function usersGet(instance: IPublicClientApplication, user: User | undefined): Promise<{ users: string[], error: string }> {\r\n // need a logged in user to get graph users\r\n if (user == null || user.spacode == \"\") {\r\n return { users: [], error: `500: invalid user passed to usersGet` };\r\n }\r\n // make /users endpoint call\r\n try {\r\n // create headers\r\n const headers = await graphDefineHeaders(instance, user);\r\n let options = { method: \"GET\", headers: headers };\r\n let usersEndpoint = getGraphEndpoint(user.authority);\r\n usersEndpoint += graphConfig.graphUsersPredicate;\r\n let response = await fetch(usersEndpoint, options);\r\n let data = await response.json();\r\n if (typeof data.error !== \"undefined\") {\r\n return { users: [], error: `${data.error.code}: ${data.error.message}` };\r\n }\r\n let users = new Array<string>();\r\n for (let user of data.value) {\r\n users.push(user.mail);\r\n }\r\n return { users: users, error: `` };\r\n }\r\n catch (error: any) {\r\n console.log(error);\r\n return { users: [], error: `Exception: ${error}` };\r\n }\r\n}\r\n// ======================= Mindline SyncConfig API ===============================\r\nexport async function auditConfigAdd(instance: IPublicClientApplication, user: User, ac: AuditConfig, debug: boolean): Promise<APIResult> {\r\n return auditConfigPost(instance, user, ac, debug);\r\n}\r\nexport async function auditConfigRetrieve(instance: IPublicClientApplication, user: User, configurationId: string, debug: boolean): Promise<APIResult> {\r\n return auditConfigGet(instance, user, configurationId, debug);\r\n}\r\nexport async function auditConfigEdit(instance: IPublicClientApplication, user: User, configurationId: string, config: AuditConfig, debug: boolean): Promise<APIResult> {\r\n return auditConfigPut(instance, user, configurationId, config, debug);\r\n}\r\nexport async function auditConfigRemove(instance: IPublicClientApplication, user: User, configurationId: string, debug: boolean): Promise<APIResult> {\r\n return auditConfigDelete(instance, user, configurationId, debug);\r\n}\r\nexport async function auditEventsRetrieve(instance: IPublicClientApplication, authorizedUser: User, debug: boolean): Promise<APIResult> {\r\n return auditEventsGet(instance, authorizedUser, debug);\r\n}\r\nexport async function configEdit(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n config: SyncConfig,\r\n setConfigId: (id: string) => void,\r\n setSelectedConfigs: (selectedConfigs: { [id: string]: boolean | number[] }) => void,\r\n workspace: Workspace,\r\n ii: InitInfo,\r\n debug: boolean): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (config.id === \"1\") {\r\n result = await configPost(instance, authorizedUser, config, workspace.id, debug);\r\n if (result.result) {\r\n // config id was updated from \"1\"\r\n if (setConfigId) {\r\n setConfigId(config.id);\r\n }\r\n // set this config as the selected config\r\n const newSelection = {};\r\n Object.defineProperty(newSelection, config.id, { value: true, writable: true, enumerable: true });\r\n setSelectedConfigs(newSelection);\r\n // update associated configs in this workspace\r\n let idx: number = workspace.associatedConfigs.findIndex((id: string) => id == \"1\");\r\n if (idx !== -1) {\r\n workspace.associatedConfigs.splice(idx, 1);\r\n workspace.associatedConfigs.push(config.id);\r\n }\r\n }\r\n }\r\n else {\r\n result = await configPut(instance, authorizedUser, config, debug);\r\n }\r\n ii.save();\r\n return result;\r\n}\r\nexport async function configEnable(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n configurationId: string,\r\n enabled: boolean,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n result = await configPatch(instance, authorizedUser, configurationId, enabled, debug);\r\n return result;\r\n}\r\nexport async function configConsentForRead(instance: IPublicClientApplication, authorizedUser: User, configId: string, tid: string, consent: boolean): Promise<APIResult> {\r\n return configConsentReadPut(instance, authorizedUser, configId, tid, consent);\r\n}\r\nexport async function configConsentForWrite(instance: IPublicClientApplication, authorizedUser: User, configId: string, tid: string, consent: boolean): Promise<APIResult> {\r\n return configConsentWritePut(instance, authorizedUser, configId, tid, consent);\r\n}\r\nexport async function configsRefresh(instance: IPublicClientApplication, authorizedUser: User, workspaceId: string, ii: InitInfo, debug: boolean, currentConfig: SyncConfig): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (debug) debugger;\r\n try {\r\n let workspace: Workspace | undefined = ii.ws.find((w) => w.id === workspaceId);\r\n if (workspace != null) {\r\n // clear SyncConfig associations as we are about to reset\r\n workspace.associatedConfigs.length = 0;\r\n // GET configs associated with this workspace\r\n let result: APIResult = await configGet(instance, authorizedUser, currentConfig, workspace.id, debug);\r\n if (!result.result) return result;\r\n // process returned workspace components\r\n // let configSelectedId: string = \"\";\r\n // for (let cid of workspace.associatedConfigs) {\r\n // let config = ii.cs.find((c: SyncConfig) => c.id === cid);\r\n // if (config != null && config.sel) {\r\n // ii.cs.find((c: SyncConfig) => c.id === cid) ??\r\n // ii.acs.find((a: AuditConfig) => a.id === cid);\r\n\r\n // if (config && config.sel) {\r\n // configSelectedId = config.id;\r\n // }\r\n // }\r\n\r\n // // Pass combined list to processor\r\n // processReturnedConfigs(workspace, ii, result.array!, currentConfig.id);\r\n // console.log(\"Init Info-----------\", ii)\r\n // console.log(\"Init Info-----------\", configSelectedId)\r\n // // tag components with workspaceIDs\r\n // ii.tagWithWorkspaces();\r\n // localStorage.setItem(\"BatchIdArray\", \"{}\");\r\n // }\r\n // processReturnedConfigs(workspace, ii, result.array!, currentConfig.id);\r\n // console.log(\"Init Info-----------\", ii)\r\n // console.log(\"Init Info-----------\", currentConfig.id)\r\n // tag components with workspaceIDs\r\n ii.tagWithWorkspaces();\r\n localStorage.setItem(\"BatchIdArray\", \"{}\");\r\n return result;\r\n } else {\r\n // workspace not found\r\n result.result = false;\r\n result.error = \"Workspace not found.\";\r\n result.status = 404;\r\n return result;\r\n }\r\n\r\n } catch (error: any) {\r\n console.log(error.message);\r\n result.error = error.message;\r\n result.result = false;\r\n result.status = 500;\r\n return result;\r\n }\r\n}\r\n\r\nexport async function configRemove(instance: IPublicClientApplication, authorizedUser: User, config: SyncConfig, workspaceId: string, debug: boolean): Promise<APIResult> {\r\n return configDelete(instance, authorizedUser, config, workspaceId, debug);\r\n}\r\nexport async function initGet(instance: IPublicClientApplication, user: User, ii: InitInfo, tasks: TaskArray, debug: boolean): Promise<APIResult> {\r\n console.log(`>>>>>> initGet`);\r\n let result: APIResult = new APIResult();\r\n if (debug)\r\n debugger;\r\n // lookup authority for this user (the lookup call does it based on domain, but TID works as well to find authority)\r\n let tenant: Tenant = new Tenant();\r\n tenant.tid = user.tid;\r\n tenant.domain = user.tid;\r\n let bResult: boolean = await tenantUnauthenticatedLookup(tenant, debug);\r\n if (bResult) {\r\n // success, we now know cloud instance where this tenant is provisioned\r\n user.authority = tenant.authority;\r\n // get tenant name and domain from AAD to pass to Configuration API\r\n tasks.setTaskStart(\"GET tenant details\", new Date());\r\n result.result = await tenantRelationshipsGetById(user, tenant, ii, instance, debug);\r\n tasks.setTaskEnd(\"GET tenant details\", new Date(), \"complete\");\r\n // store tenant name and domain on user, POST user and tenant to back end\r\n if (result.result) {\r\n user.companyName = tenant.name;\r\n user.companyDomain = tenant.domain;\r\n tasks.setTaskStart(\"POST config init\", new Date());\r\n result = await initPost(instance, user, debug);\r\n tasks.setTaskEnd(\"POST config init\", new Date(), result.result ? \"complete\" : \"failed\");\r\n }\r\n // simlarly, if we just did our first post, then query config backend for workspace(s) associated with this user\r\n if (result.result) {\r\n tasks.setTaskStart(\"GET workspaces\", new Date());\r\n result = await workspaceInfoGet(instance, user, ii, debug);\r\n tasks.setTaskEnd(\"GET workspaces\", new Date(), result.result ? \"complete\" : \"failed\");\r\n }\r\n console.log(\"initGet complete. Version: \" + getSyncVersion());\r\n return result;\r\n }\r\n else {\r\n result.error = `Failed to retrieve authority for user \"${user.mail}\" TID ${user.tid}.`;\r\n result.result = false;\r\n return result;\r\n }\r\n}\r\nexport async function tenantAdd(instance: IPublicClientApplication, authorizedUser: User, tenant: Tenant, workspaceId: string): Promise<APIResult> {\r\n return tenantPost(instance, authorizedUser, tenant, workspaceId);\r\n}\r\nexport async function tenantRemove(instance: IPublicClientApplication, authorizedUser: User, tenant: Tenant, workspaceId: string, debug: boolean): Promise<APIResult> {\r\n return tenantDelete(instance, authorizedUser, tenant, workspaceId, debug);\r\n}\r\nexport async function userAdd(instance: IPublicClientApplication, authorizedUser: User, user: User, workspaceId: string): Promise<APIResult> {\r\n return adminPost(instance, authorizedUser, user, workspaceId);\r\n}\r\nexport async function userRemove(instance: IPublicClientApplication, authorizedUser: User, user: User, workspaceId: string): Promise<APIResult> {\r\n return adminDelete(instance, authorizedUser, user, workspaceId);\r\n}\r\nexport async function workspaceEdit(instance: IPublicClientApplication, authorizedUser: User, workspaceId: string, workspaceName: string): Promise<APIResult> {\r\n return await workspacePut(instance, authorizedUser, workspaceId, workspaceName);\r\n}\r\n// retrieve Workspace(s), User(s), Tenant(s), SyncConfig(s) given newly logged in user\r\nfunction processReturnedAdmins(workspace: Workspace, ii: InitInfo, returnedAdmins: Array<Object>, adminSelectedId: string) {\r\n returnedAdmins.map((item: any) => {\r\n // are we already tracking this user?\r\n let user: User | null | undefined = null;\r\n let usIndex = ii.us.findIndex((u) => (u.oid === item.userId || u.oid === item.email));\r\n if (usIndex === -1) {\r\n // start tracking\r\n let dummyIndex = ii.us.findIndex((u) => u.oid === \"1\");\r\n if (dummyIndex !== -1) {\r\n // clear and overwrite dummy\r\n user = ii.us.at(dummyIndex);\r\n // replace dummy User oid \"1\" with real oid in associatedTenants of this workspace\r\n let idx: number = workspace.associatedUsers.findIndex((oid: string) => oid == \"1\");\r\n if (idx !== -1) {\r\n workspace.associatedUsers.splice(idx, 1);\r\n workspace.associatedUsers.push(item.userId);\r\n }\r\n }\r\n else {\r\n // create and track new user\r\n user = new User();\r\n ii.us.push(user);\r\n }\r\n } else {\r\n // already tracking this user\r\n user = ii.us.at(usIndex);\r\n }\r\n // restore selection\r\n user!.sel = (adminSelectedId === item.userId);\r\n // refresh all the data available from the server\r\n user!.oid = item.userId ? item.userId : item.email;\r\n user!.name = item.firstName ?? user!.name;\r\n user!.mail = item.email;\r\n user!.tid = item.tenantId;\r\n // *try* to set authority/companyName/companyDomain from tenant returned in previous call processReturnedTenants (it may not be there)\r\n // ASSUMPTION: in terms of setting authority, user either comes from\r\n // 1. .NET session, in which case user has an authority from the token\r\n // 2. SyncConfig API, in which case either\r\n // a. associated tenant with authority exists for this user (i.e. workspace owner has called POST init/configuration and created Admin and Tenant at same time)\r\n // b. user has never logged in (i.e. workspace owner has invited an incomplete admin to the workspace and there is no corresponding tenant)\r\n // i. this means that we *may* not have an authority stored for this user\r\n let tenant: Tenant | undefined = ii.ts.find((t) => (t.tid === user!.tid));\r\n if (tenant != undefined) {\r\n user!.authority = tenant.authority;\r\n user!.companyName = tenant.name;\r\n user!.companyDomain = tenant.domain;\r\n }\r\n // ensure this workspace tracks this user\r\n let idx = workspace.associatedUsers.findIndex((u) => u === user!.oid);\r\n if (idx == -1) workspace.associatedUsers.push(user!.oid);\r\n });\r\n ii.save();\r\n}\r\nfunction processReturnedTenants(workspace: Workspace, ii: InitInfo, returnedTenants: Array<Object>, tenantSelectedId: string) {\r\n returnedTenants.map((item: any) => {\r\n // are we already tracking this tenant?\r\n let tenant: Tenant | null | undefined = null;\r\n let tsIndex = ii.ts.findIndex((t) => t.tid === item.tenantId);\r\n if (tsIndex === -1) {\r\n // start tracking\r\n let dummyIndex = ii.ts.findIndex((t) => t.tid === \"1\");\r\n if (dummyIndex !== -1) {\r\n // clear and overwrite dummy\r\n tenant = ii.ts.at(dummyIndex);\r\n // replace dummy Tenant id \"1\" with real id in associatedTenants of this workspace\r\n let idx: number = workspace.associatedTenants.findIndex((tid: string) => tid == \"1\");\r\n if (idx !== -1) {\r\n workspace.associatedTenants.splice(idx, 1);\r\n workspace.associatedTenants.push(item.tenantId);\r\n }\r\n } else {\r\n // create and track new tenant\r\n tenant = new Tenant();\r\n ii.ts.push(tenant);\r\n }\r\n } else {\r\n // already tracking this tenant\r\n tenant = ii.ts.at(tsIndex);\r\n }\r\n // restore selection\r\n tenant!.sel = (tenantSelectedId === item.tenantId);\r\n // refresh all the data available from the server\r\n tenant!.tid = item.tenantId;\r\n tenant!.name = item.name;\r\n tenant!.domain = item.domain;\r\n tenant!.tenantType = item.type.toLowerCase(); // should now be strings\r\n\r\n // canonicalize authority when getting it from config backend\r\n const regex = /^(https:\\/\\/login.microsoftonline.(?:us|com)\\/)organizations\\/v2.0$/;\r\n const regexMatch = item.authority.match(regex);\r\n tenant!.authority = regexMatch ? regexMatch[1] : item.authority;\r\n\r\n // ensure this workspace tracks this tenant\r\n let idx = workspace.associatedTenants.findIndex((t) => t === item.tenantId);\r\n if (idx == -1) workspace.associatedTenants.push(item.tenantId);\r\n });\r\n ii.save();\r\n}\r\nfunction processReturnedConfigs(workspace: Workspace, ii: InitInfo, returnedConfigs: Array<Object>, configSelectedId: string) {\r\n // process returned configs\r\n returnedConfigs.map((item: any) => {\r\n // are we already tracking this config?\r\n let config: SyncConfig | null | undefined = null;\r\n let csIndex = ii.cs.findIndex((c) => c.id === item.id);\r\n if (csIndex === -1) {\r\n // start tracking\r\n let dummyIndex = ii.cs.findIndex((c) => c.id === \"1\");\r\n if (dummyIndex !== -1) {\r\n // clear and overwrite dummy\r\n config = ii.cs.at(dummyIndex);\r\n // replace dummy SyncConfig id \"1\" with real id in associatedConfigs of this workspace\r\n let idx: number = workspace.associatedConfigs.findIndex((id: string) => id == \"1\");\r\n if (idx !== -1) {\r\n workspace.associatedConfigs.splice(idx, 1);\r\n workspace.associatedConfigs.push(item.id);\r\n }\r\n } else {\r\n // create and track new workspace\r\n config = new SyncConfig();\r\n ii.cs.push(config);\r\n }\r\n } else {\r\n // already tracking this config\r\n config = ii.cs.at(csIndex);\r\n }\r\n // restore selection\r\n config!.sel = (configSelectedId === item.id);\r\n // refresh all the data available from the server\r\n config!.id = item.id;\r\n config!.workspaceId = item.workspaceId;\r\n config!.name = item.name;\r\n config!.description = item.description;\r\n config!.isEnabled = item.isEnabled;\r\n // create TenantConfigInfo array\r\n config!.tenants.length = 0;\r\n item.tenants.map((tci: any) => {\r\n let tenantConfigInfo = new TenantConfigInfo();\r\n tenantConfigInfo.tid = tci.tenantId;\r\n tenantConfigInfo.sourceGroupId = tci.sourceGroupId ?? \"\";\r\n tenantConfigInfo.sourceGroupName = tci.sourceGroupName ?? \"\";\r\n tenantConfigInfo.targetGroupId = tci.targetGroupId ?? \"\";\r\n tenantConfigInfo.targetGroupName = tci.targetGroupName ?? \"\";\r\n tenantConfigInfo.configurationTenantType = tci.configurationTenantType.toLowerCase();\r\n tenantConfigInfo.deltaToken = tci.deltaToken ?? \"\";\r\n tenantConfigInfo.configId = config!.id;\r\n tenantConfigInfo.batchId = tci.batchId ?? \"\";\r\n tenantConfigInfo.isReadPermissionConsented = tci.isReadPermissionConsented;\r\n tenantConfigInfo.isWritePermissionConsented = tci.isWritePermissionConsented;\r\n config!.tenants.push(tenantConfigInfo);\r\n });\r\n // ensure this workspace tracks this config\r\n let idx = workspace.associatedConfigs.findIndex((c) => c === item.id);\r\n if (idx == -1) workspace.associatedConfigs.push(item.id);\r\n });\r\n console.log(\"--------------ii-----------\", ii)\r\n ii.save();\r\n}\r\nasync function workspaceInfoGet(instance: IPublicClientApplication, user: User, ii: InitInfo, debug: boolean): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (debug) debugger;\r\n try {\r\n result = await workspacesGet(instance, user, debug);\r\n if (result.result) {\r\n for (let o of result.array! as any[]) {\r\n // are we already tracking this workspace?\r\n let workspace: Workspace | null | undefined = null;\r\n let wsIndex = ii.ws.findIndex((w) => w.id === o.id);\r\n if (wsIndex === -1) {\r\n // start tracking\r\n let dummyIndex = ii.ws.findIndex((w) => w.id === \"1\");\r\n if (dummyIndex !== -1) {\r\n // clear and overwrite dummy\r\n workspace = ii.ws.at(dummyIndex);\r\n }\r\n else {\r\n // create and track new workspace\r\n workspace = new Workspace();\r\n ii.ws.push(workspace);\r\n }\r\n } else {\r\n // already tracking this workspace\r\n workspace = ii.ws.at(wsIndex);\r\n }\r\n // preserve selected admin, tenant, and config\r\n let adminSelectedId: string = \"\";\r\n for (let oid of workspace!.associatedUsers) {\r\n let user = ii.us.find((u: User) => u.oid === oid);\r\n if (user != null && user.sel) {\r\n adminSelectedId = user.oid;\r\n }\r\n }\r\n let tenantSelectedId: string = \"\";\r\n for (let tid of workspace!.associatedTenants) {\r\n let tenant = ii.ts.find((t: Tenant) => t.tid === tid);\r\n if (tenant != null && tenant.sel) {\r\n tenantSelectedId = tenant.tid;\r\n }\r\n }\r\n let configSelectedId: string = \"\";\r\n for (let cid of workspace!.associatedConfigs) {\r\n let config = ii.cs.find((c: SyncConfig) => c.id === cid);\r\n if (config != null && config.sel) {\r\n configSelectedId = config.id;\r\n }\r\n }\r\n // clear associations as we are about to reset\r\n workspace!.associatedUsers.length = 0;\r\n workspace!.associatedTenants.length = 0;\r\n workspace!.associatedConfigs.length = 0;\r\n // set id and name based on returned data\r\n workspace!.id = o.id;\r\n workspace!.name = o.name;\r\n workspace!.ownerid = o.workspaceOwnerUserId;\r\n // parallel GET admins, tenants, configs associated with this workspace\r\n let adminsPromise: Promise<APIResult> = adminsGet(instance, user, workspace!.id, debug);\r\n let tenantsPromise: Promise<APIResult> = tenantsGet(instance, user, workspace!.id, debug);\r\n let configsPromise: Promise<APIResult> = configsGet(instance, user, workspace!.id, debug);\r\n // wait for all to finish, return on any failure\r\n let [adminsResult, tenantsResult, configsResult] = await Promise.all([adminsPromise, tenantsPromise, configsPromise]);\r\n if (!adminsResult.result) return adminsResult;\r\n if (!tenantsResult.result) return tenantsResult;\r\n if (!configsResult.result) return configsResult;\r\n // process returned workspace components (tenants first as they have the authorities that admins depend on)\r\n processReturnedTenants(workspace!, ii, tenantsResult.array!, tenantSelectedId);\r\n processReturnedAdmins(workspace!, ii, adminsResult.array!, adminSelectedId);\r\n processReturnedConfigs(workspace!, ii, configsResult.array!, configSelectedId);\r\n // tag components with workspaceIDs\r\n ii.tagWithWorkspaces();\r\n }\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(error.message);\r\n result.error = error.message;\r\n }\r\n result.result = false;\r\n result.status = 500;\r\n return result;\r\n}\r\n// ======================= PowerBI Access Token Retrieval ===============================\r\nexport async function getPowerBIAccessToken(\r\n instance: IPublicClientApplication,\r\n user: User\r\n): Promise<string> {\r\n let accesstoken: string = \"\";\r\n let syncversion: string = getSyncVersion();\r\n // retrieve access token silently\r\n try {\r\n let accounts: AccountInfo[] = instance.getAllAccounts();\r\n let homeAccountId = user.oid + \".\" + user.tid;\r\n let account: AccountInfo | null = null;\r\n for (let i: number = 0; i < accounts.length; i++) {\r\n if (accounts[i].homeAccountId == homeAccountId) {\r\n account = accounts[i];\r\n }\r\n }\r\n let response: AuthenticationResult = await instance.acquireTokenSilent({\r\n scopes: [\"https://analysis.windows.net/powerbi/api/App.Read.All\",\r\n \"https://analysis.windows.net/powerbi/api/Dataset.ReadWrite.All\",\r\n \"https://analysis.windows.net/powerbi/api/Report.ReadWrite.All\",\r\n \"https://analysis.windows.net/powerbi/api/Workspace.ReadWrite.All\"\r\n ],\r\n account: account!\r\n });\r\n accesstoken = response.accessToken; // cache access token\r\n console.log(\"PowerBI token acquired silently: \" + accesstoken.slice(0, 20));\r\n }\r\n catch (error: any) {\r\n console.log(\"PowerBI token failed to be acquired silently\");\r\n }\r\n console.log(syncversion);\r\n return accesstoken;\r\n}\r\n// ======================= Azure REST API ===============================\r\n// TODO: this is where you want to trigger a re-authentication if token expires\r\nasync function azureDefineHeaders(\r\n instance: IPublicClientApplication,\r\n user: User\r\n): Promise<Headers> {\r\n const headers = new Headers();\r\n headers.append(\"Content-Type\", \"application/json\");\r\n headers.append(\"accept\", \"*/*\");\r\n // authorization header - if needed, retrieve and cache access token\r\n if (user.azureAccessToken == null || user.azureAccessToken === \"\") {\r\n try {\r\n let accounts: AccountInfo[] = instance.getAllAccounts();\r\n let homeAccountId = user.oid + \".\" + user.tid;\r\n let account: AccountInfo | null = null;\r\n for (let i: number = 0; i < accounts.length; i++) {\r\n if (accounts[i].homeAccountId == homeAccountId) {\r\n account = accounts[i];\r\n }\r\n }\r\n let response: AuthenticationResult = await instance.acquireTokenSilent({\r\n scopes: [\"https://management.azure.com/user_impersonation\"],\r\n account: account!\r\n });\r\n user.azureAccessToken = response.accessToken; // cache access token\r\n console.log(\"Front end token acquired silently: \" + user.azureAccessToken.slice(0, 20));\r\n }\r\n catch (error: any) {\r\n try {\r\n console.log(\"Front end token silent acquisition failure: \" + error);\r\n // fallback to redirect if silent acquisition fails\r\n let accounts: AccountInfo[] = instance.getAllAccounts();\r\n let homeAccountId = user.oid + \".\" + user.tid;\r\n let account: AccountInfo | null = null;\r\n for (let i: number = 0; i < accounts.length; i++) {\r\n if (accounts[i].homeAccountId == homeAccountId) {\r\n account = accounts[i];\r\n }\r\n }\r\n instance.acquireTokenRedirect({\r\n scopes: [\"https://management.azure.com/user_impersonation\"],\r\n account: account!\r\n });\r\n }\r\n catch (error: any) {\r\n console.log(\"Front end token popup acquisition failure: \" + error);\r\n }\r\n }\r\n }\r\n headers.append(\"Authorization\", `Bearer ${user.azureAccessToken}`);\r\n return headers;\r\n}\r\nexport async function canListRootAssignments(instance: IPublicClientApplication, user: User): Promise<boolean> {\r\n // need a logged in user to call Azure REST API\r\n if (user == null || user.spacode == \"\") {\r\n return false;\r\n }\r\n // make Azure REST API call\r\n try {\r\n // create headers\r\n const headers = await azureDefineHeaders(instance, user);\r\n let options = { method: \"GET\", headers: headers };\r\n let listrootassignmentsEndpoint: string = azureConfig.azureListRootAssignments;\r\n listrootassignmentsEndpoint += \"'\";\r\n listrootassignmentsEndpoint += user.oid;\r\n listrootassignmentsEndpoint += \"'\";\r\n let response = await fetch(listrootassignmentsEndpoint, options);\r\n if (response.status == 200) {\r\n let data: any = await response.json();\r\n data = data;\r\n debugger;\r\n console.log(\"Successful call to Azure Resource Graph list root assignments\");\r\n }\r\n else {\r\n console.log(await processErrors(response));\r\n return false;\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(error);\r\n return false;\r\n }\r\n return true;\r\n}\r\nexport async function elevateGlobalAdminToUserAccessAdmin(instance: IPublicClientApplication, user: User): Promise<boolean> {\r\n // need a logged in user to call Azure REST API\r\n if (user == null || user.spacode == \"\") {\r\n return false;\r\n }\r\n // make Azure REST API call\r\n try {\r\n // create headers\r\n const headers = await azureDefineHeaders(instance, user);\r\n let options = { method: \"POST\", headers: headers };\r\n let elevateaccessEndpoint: string = azureConfig.azureElevateAccess;\r\n let response = await fetch(elevateaccessEndpoint, options);\r\n if (response.status == 200) {\r\n console.log(\"Successful call to Azure Resource Graph list root assignments\");\r\n }\r\n else {\r\n console.log(await processErrors(response));\r\n return false;\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(error);\r\n return false;\r\n }\r\n return true;\r\n}\r\nasync function readResources(instance: IPublicClientApplication, user: User): Promise<ResourceNode[]> {\r\n // need a logged in user to call Azure REST API\r\n let resources: ResourceNode[] = new Array<ResourceNode>();\r\n if (user == null || user.spacode == \"\") {\r\n return resources;\r\n }\r\n // make Azure REST API call\r\n try {\r\n // create headers\r\n const headers = await azureDefineHeaders(instance, user);\r\n let options = { method: \"GET\", headers: headers };\r\n let listrootassignmentsEndpoint: string = azureConfig.azureListRootAssignments;\r\n listrootassignmentsEndpoint += \"'\";\r\n listrootassignmentsEndpoint += user.oid;\r\n listrootassignmentsEndpoint += \"'\";\r\n let response = await fetch(listrootassignmentsEndpoint, options);\r\n if (response.status == 200) {\r\n let data = await response.json();\r\n data = data;\r\n debugger;\r\n console.log(\"Successful call to Azure Resource Graph list root assignments\");\r\n }\r\n else {\r\n console.log(await processErrors(response));\r\n return resources;\r\n }\r\n }\r\n catch (error: any) {\r\n console.log(error);\r\n return resources;\r\n }\r\n return resources;\r\n}\r\n// ======================= HYBRIDSPA.TS -- Mindline SyncConfig API helper functions ===============================\r\nfunction getAPIScope(user: User): string {\r\n let apiAppID: string = \"8d95d21c-c378-4bb0-9f52-88c30d271e7a\";\r\n let authority: string = user.authority.toLowerCase();\r\n if (authority.startsWith(\"https://login.microsoftonline.com/\"))\r\n apiAppID = \"8d95d21c-c378-4bb0-9f52-88c30d271e7a\";\r\n else if (authority.startsWith(\"https://login.microsoftonline.us/\"))\r\n apiAppID = \"48da942e-ea3d-49e4-a054-81649012f8f2\";\r\n else if (authority.startsWith(\"https://login.partner.microsoftonline.cn/\"))\r\n apiAppID = \"c91d32e4-dcc5-4d77-826a-16e93ffce666\";\r\n let apiScope: string = `api://${apiAppID}/Config.Write`;\r\n return apiScope;\r\n}\r\n// TODO: this is where you want to trigger a re-authentication if token expires\r\nasync function mindlineDefineHeaders(\r\n instance: IPublicClientApplication,\r\n user: User\r\n): Promise<Headers> {\r\n const headers = new Headers();\r\n headers.append(\"Content-Type\", \"application/json\");\r\n headers.append(\"accept\", \"*/*\");\r\n // always call acquireTokenSilent, handling expired tokens on exception\r\n const apiScope: string = getAPIScope(user);\r\n try {\r\n let accounts: AccountInfo[] = instance.getAllAccounts();\r\n let homeAccountId = user.oid + \".\" + user.tid;\r\n let account: AccountInfo | null = null;\r\n for (let i: number = 0; i < accounts.length; i++) {\r\n if (accounts[i].homeAccountId == homeAccountId) {\r\n account = accounts[i];\r\n }\r\n }\r\n let response: AuthenticationResult = await instance.acquireTokenSilent({\r\n scopes: [apiScope],\r\n account: account!\r\n });\r\n user.mindlineAccessToken = response.accessToken; // cache access token\r\n console.log(\"Front end mindline token acquired silently: \" + user.mindlineAccessToken.slice(0, 20));\r\n }\r\n catch (error: any) {\r\n try {\r\n console.log(\"Front end mindline token silent acquisition failure, triggering redirect: \" + error);\r\n // fallback to redirect if silent acquisition fails\r\n let accounts: AccountInfo[] = instance.getAllAccounts();\r\n let homeAccountId = user.oid + \".\" + user.tid;\r\n let account: AccountInfo | null = null;\r\n for (let i: number = 0; i < accounts.length; i++) {\r\n if (accounts[i].homeAccountId == homeAccountId) {\r\n account = accounts[i];\r\n }\r\n }\r\n // assumption: this redirect will trigger login flow callbacks in program.cs\r\n instance.acquireTokenRedirect({\r\n scopes: [apiScope],\r\n account: account!\r\n });\r\n }\r\n catch (error: any) {\r\n console.log(\"Front end mindline token redirect acquisition failure: \" + error);\r\n }\r\n }\r\n headers.append(\"Authorization\", `Bearer ${user.mindlineAccessToken}`);\r\n return headers;\r\n}\r\nexport async function processErrors(response: Response): Promise<string> {\r\n let errorString: string = \"\";\r\n if (response.status === 401) {\r\n errorString = response.statusText;\r\n if (errorString != \"\") return errorString;\r\n }\r\n let data = await response.json();\r\n // process errors from Mindline SyncConfig API\r\n if (data.error !== undefined) {\r\n errorString = `Error: ${data.error} Message: ${data.message}`;\r\n } else if (data.errors !== undefined) {\r\n let errorArray = Object.keys(data.errors);\r\n let errorlist: string = \"\";\r\n errorString = errorArray.reduce(\r\n (acc, curr) => acc + curr + \": \" + data.errors[curr] + \" \",\r\n errorlist\r\n );\r\n } else if (data.title !== undefined) {\r\n errorString = data.title;\r\n } else {\r\n debugger;\r\n }\r\n return errorString;\r\n}\r\n//adminDelete\r\nexport async function adminDelete(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n user: User,\r\n workspaceId: string\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we need either oid or mail for the user and valid workspace id\r\n if ((user.oid == \"\" && user.mail == \"\") || workspaceId == \"\") {\r\n result.result = false;\r\n result.error = \"adminDelete: invalid parameters\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // define admin endpoint and add workspaceId parameter\r\n let url: URL | null = null;\r\n url = new URL(mindlineConfig.adminEndpoint());\r\n url.searchParams.append(\"workspaceId\", workspaceId);\r\n url.searchParams.append(\"email\", user.mail);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // make endpoint call\r\n let options = { method: \"DELETE\", headers: headers };\r\n try {\r\n console.log(\"Attempting DELETE from /admin: \" + url!.href);\r\n let response = await fetch(url!.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful DELETE from /admin: ${url!.href}`);\r\n return result;\r\n } else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed DELETE from /admin: ${url.href}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n } catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//adminsGet\r\nexport async function adminsGet(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n workspaceID: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we need a workspace id\r\n if (workspaceID === \"\") {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"adminsGet: no workspace provided\";\r\n return result;\r\n }\r\n // create endpoint\r\n let endpoint: string = mindlineConfig.adminsEndpoint();\r\n // add parameter to endpoint\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"workspaceId\", workspaceID);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n // make endpoint call\r\n let options = { method: \"GET\", headers: headers };\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting GET from /admins: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let returnedArray: Array<Object> = await response.json();\r\n if (returnedArray != null) {\r\n result.array = returnedArray;\r\n let initialValue: string = \"\";\r\n console.log(`Successful GET from /admins: ${result.array.reduce((acc, curr: any) => acc + curr.email + \" \", initialValue)}`);\r\n return result;\r\n }\r\n else {\r\n result.error = `Failed GET from /admins: failed to JSON-parse response`;\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n else {\r\n console.log(`Failed GET from /admins: ${url.href}`);\r\n result.error = await processErrors(response);\r\n result.status = 500;\r\n result.result = false;\r\n console.log(result.error);\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//adminPost: write validated admin to back end\r\nexport async function adminPost(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n user: User,\r\n workspaceId: string\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (user.mail == \"\" || user.authority == \"\" || user.tid === \"\") {\r\n result.result = false;\r\n result.error = \"adminPost: invalid argument\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create admin endpoint\r\n let endpoint: string = mindlineConfig.adminEndpoint();\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // create admin body\r\n let adminBody: string = `\r\n {\"email\": \"${user.mail}\", \r\n \"tenantId\": \"${user.tid}\", \r\n \"workspaceId\": \"${workspaceId}\"\r\n }`;\r\n let options = { method: \"POST\", headers: headers, body: adminBody };\r\n // make admin endpoint call\r\n try {\r\n console.log(\"Attempting POST to /admin: \" + endpoint);\r\n let response = await fetch(endpoint, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful POST to /admin: ${adminBody}`);\r\n return result;\r\n } else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed POST to /admin: ${adminBody}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n } catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n\r\n//auditConfigPost: write audit config to back end\r\nexport async function auditConfigPost(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n config: AuditConfig,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n instance = instance;\r\n user = user;\r\n debug = debug;\r\n let result: APIResult = new APIResult();\r\n if (config.id === \"\") {\r\n result.result = false;\r\n result.error = \"auditConfigPost: invalid config ID\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create no parameter audit config post endpoint\r\n let endpoint: string = mindlineConfig.auditConfigEndpoint();\r\n // create config headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n // create config body\r\n let configBody: string = `\r\n {\r\n \"workspaceId\": \"${config.workspaceId}\",\r\n \"name\": \"${config.name}\",\r\n \"description\": \"${config.description}\",\r\n \"isEnabled\": ${config.isEnabled},\r\n \"email\": \"${config.email}\",\r\n \"tenantId\": \"${config.tenantId}\"\r\n }`;\r\n let options = { method: \"POST\", headers: headers, body: configBody };\r\n // make config endpoint call\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting POST to /audit-configuration: \" + endpoint);\r\n let response = await fetch(endpoint, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let data = await response.json();\r\n config.id = data;\r\n console.log(\r\n `Successful AuditConfigID: ${data} from POST to /audit-configuration: ${configBody}`\r\n );\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed POST to /audit-configuration: ${configBody}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.status = 500;\r\n result.result = false;\r\n result.error = error.message;\r\n console.log(result.error);\r\n return result;\r\n }\r\n return result;\r\n}\r\n\r\n//auditConfigGET: get audit config from back end\r\nexport async function auditConfigGet(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n configurationId: string,\r\n debug: boolean): Promise<APIResult> {\r\n instance = instance;\r\n user = user;\r\n debug = debug;\r\n\r\n let result: APIResult = new APIResult();\r\n\r\n // Validate user\r\n if (user.mail == undefined || user.mail == \"\") {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"auditConfigGet: no user provided\";\r\n return result;\r\n }\r\n\r\n //Validate input\r\n if (!configurationId || configurationId.trim() === \"\") {\r\n result.result = false;\r\n result.error = \"auditConfigGet: configurationId is required\";\r\n result.status = 500;\r\n return result;\r\n }\r\n\r\n //Define endpoint with query parameter (configurationId)\r\n let endpoint: string = mindlineConfig.auditConfigEndpoint();\r\n let url: URL = new URL(`${endpoint}/${configurationId}`);\r\n\r\n //Create headers with Entra ID token\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n\r\n let options = { method: \"GET\", headers: headers };\r\n\r\n //Make the GET request\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting GET to /audit-configuration: \" + url.href);\r\n\r\n const response = await fetch(url.href, options);\r\n\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let data = await response.json();\r\n result.result = true;\r\n result.status = response.status;\r\n console.log(\"Successful GET from /audit-configuration:\", data);\r\n } else {\r\n result.error = await processErrors(response);\r\n console.log(\"Failed GET from /audit-configuration: \" + url.href);\r\n console.log(result.error);\r\n result.status = response.status;\r\n result.result = false;\r\n }\r\n } catch (error: any) {\r\n result.status = 500;\r\n result.result = false;\r\n result.error = error.message;\r\n console.log(\"Error during auditConfigGet:\", error.message);\r\n }\r\n\r\n return result;\r\n}\r\n\r\n//auditConfigPUT: put audit config from back end \r\nexport async function auditConfigPut(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n configurationId: string,\r\n config: AuditConfig, // model with name, description, isEnabled, isReadPermissionConsented, email\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n\r\n // Validate configurationId\r\n if (!configurationId || configurationId.trim() === \"\") {\r\n result.result = false;\r\n result.error = \"auditConfigPut: invalid configuration ID\";\r\n result.status = 500;\r\n return result;\r\n }\r\n\r\n // Build endpoint with query parameter\r\n let endpoint: string = mindlineConfig.auditConfigEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"configurationId\", configurationId);\r\n\r\n // Create headers with token\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n\r\n // Build body \r\n let configBody: string = `\r\n {\r\n \"name\": \"${config.name}\",\r\n \"description\": \"${config.description}\",\r\n \"isEnabled\": ${config.isEnabled},\r\n \"isReadPermissionConsented\": ${config.isReadPermissionConsented},\r\n \"email\": \"${config.email}\"\r\n }`;\r\n\r\n let options = { method: \"PUT\", headers: headers, body: configBody };\r\n\r\n // Call API\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting PUT to /audit-configuration: \" + url.href);\r\n\r\n const response = await fetch(url.href, options);\r\n\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n result.result = true;\r\n console.log(`Successful PUT to ${url.href}: ${configBody}`);\r\n } else {\r\n result.error = await processErrors(response);\r\n console.error(`Failed PUT to ${url.href}: ${configBody}`);\r\n result.status = response.status;\r\n result.result = false;\r\n }\r\n } catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.error(\"Error during auditConfigPut:\", error.message);\r\n }\r\n\r\n return result;\r\n}\r\n\r\n//auditConfigDELETE: delete audit config from back end \r\nexport async function auditConfigDelete(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n configurationId: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n\r\n // Validate user\r\n if (!user.mail) {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"auditConfigDelete: no user provided\";\r\n return result;\r\n }\r\n\r\n // Validate configurationId\r\n if (!configurationId || configurationId.trim() === \"\") {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"auditConfigDelete: configurationId is required\";\r\n return result;\r\n }\r\n\r\n // Build endpoint with query parameter\r\n let endpoint: string = mindlineConfig.auditConfigEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"configurationId\", configurationId);\r\n\r\n // Build request headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n const options = { method: \"DELETE\", headers: headers };\r\n\r\n try {\r\n if (debug) debugger;\r\n console.log(`Attempting DELETE to /audit-configuration: ${url.href}`);\r\n\r\n const response = await fetch(url.href, options);\r\n\r\n if (response.ok) {\r\n result.result = true;\r\n console.log(`Successful DELETE for configurationId ${configurationId}`);\r\n } else {\r\n result.error = await processErrors(response);\r\n result.status = response.status;\r\n result.result = false;\r\n console.error(`Failed DELETE for configurationId ${configurationId}`);\r\n }\r\n } catch (error: any) {\r\n result.status = 500;\r\n result.result = false;\r\n result.error = error.message;\r\n console.error(\"Error during auditConfigDelete:\", error.message);\r\n }\r\n\r\n return result;\r\n}\r\n\r\n\r\nexport async function auditEventsGet(instance: IPublicClientApplication, user: User, debug: boolean): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we need a valid user\r\n if (user.mail == undefined || user.mail == \"\") {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"auditEventsGet: no user provided\";\r\n return result;\r\n }\r\n // create endpoint\r\n let endpoint: string = mindlineConfig.auditEventsEndpoint();\r\n let url: URL = new URL(endpoint);\r\n // create workspace headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n // make workspace endpoint call\r\n let options = { method: \"GET\", headers: headers };\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting GET from /audit-events endpoint: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let json = await response.json();\r\n if (Array.isArray(json.data)) {\r\n let returnedArray: Array<Object> = json.data;\r\n if (returnedArray != null) {\r\n result.array = returnedArray;\r\n let initialValue: string = \"\";\r\n console.log(`Successful GET from /audit-events: ${result.array.reduce((acc, curr: any) => acc + curr.initiatedByUser + \" \", initialValue)}`);\r\n return result;\r\n }\r\n }\r\n result.error = `Failed GET from /audit-events: failed to JSON-parse response`;\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n else {\r\n console.log(`Failed GET from /audit-events: ${url.href}`);\r\n result.error = await processErrors(response);\r\n result.status = 500;\r\n result.result = false;\r\n console.log(result.error);\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//configConsentReadPut\r\nexport async function configConsentReadPut(instance: IPublicClientApplication, authorizedUser: User, configId: string, tid: string, consent: boolean): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // create parameterized config consent endpoint\r\n let endpoint: string = mindlineConfig.configConsentEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"configurationId\", configId);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // create body\r\n let configConsentReadBody: string = `\r\n {\r\n \"tenantId\": \"${tid}\",\r\n \"isReadPermissionConsented\": ${consent ? \"true\" : \"false\"}\r\n }`;\r\n // make endpoint call\r\n let options = { method: \"PUT\", headers: headers, body: configConsentReadBody };\r\n try {\r\n console.log(\"Attempting PUT read consent to /configuration/consent: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful PUT to ${url.href}`);\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed PUT to ${url.href}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//configConsentWritePut\r\nexport async function configConsentWritePut(instance: IPublicClientApplication, authorizedUser: User, configId: string, tid: string, consent: boolean): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // create parameterized config consent endpoint\r\n let endpoint: string = mindlineConfig.configConsentEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"configurationId\", configId);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // create body\r\n let configConsentWriteBody: string = `\r\n {\r\n \"tenantId\": \"${tid}\",\r\n \"isWritePermissionConsented\": ${consent ? \"true\" : \"false\"}\r\n }`;\r\n // make endpoint call\r\n let options = { method: \"PUT\", headers: headers, body: configConsentWriteBody };\r\n try {\r\n console.log(\"Attempting PUT read consent to /configuration/consent: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful PUT to ${url.href}`);\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed PUT to ${url.href}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//configDelete\r\nexport async function configGet(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n config: SyncConfig,\r\n workspaceId: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n debug = debug;\r\n let result: APIResult = new APIResult();\r\n if (config.id === \"\" || workspaceId == \"\") {\r\n result.result = false;\r\n result.error = \"configGet: invalid config ID\";\r\n result.status = 500;\r\n return result;\r\n }\r\n let url: URL | null = null;\r\n url = new URL(mindlineConfig.configEndpoint());\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // make endpoint call\r\n let options = { method: \"GET\", headers: headers };\r\n try {\r\n console.log(\"Attempting GET from /config: \" + url.href);\r\n let response = await fetch(`${url.href}/${config.id}`, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful GET from /config`);\r\n let data = await response.json()\r\n return {\r\n ...result,\r\n array: [data]\r\n };\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed GET from ${url.href}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//configDelete\r\nexport async function configDelete(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n config: SyncConfig,\r\n workspaceId: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n debug = debug;\r\n let result: APIResult = new APIResult();\r\n if (config.id === \"\" || workspaceId == \"\") {\r\n result.result = false;\r\n result.error = \"configPost: invalid config ID\";\r\n result.status = 500;\r\n return result;\r\n }\r\n let url: URL | null = null;\r\n url = new URL(mindlineConfig.configEndpoint());\r\n url.searchParams.append(\"configurationId\", config.id);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // make endpoint call\r\n let options = { method: \"DELETE\", headers: headers };\r\n try {\r\n console.log(\"Attempting DELETE from /config: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful DELETE from /config`);\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed DELETE from ${url.href}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//configPatch\r\nexport async function configPatch(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n configurationId: string,\r\n enabled: boolean,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (configurationId === \"\") {\r\n result.result = false;\r\n result.error = \"configPatch: invalid config ID\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create parametrized config endpoint\r\n let endpoint: string = mindlineConfig.configEnabledEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"configurationId\", configurationId);\r\n url.searchParams.append(\"isEnabled\", enabled.toString());\r\n // create config headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n let options = { method: \"PATCH\", headers: headers };\r\n // make config endpoint call\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting PATCH to /config: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful PATCH to ${url.href}: ${enabled.toString()}`);\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed PATCH to ${url.href}: ${enabled.toString()}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//configPost\r\nexport async function configPost(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n config: SyncConfig,\r\n workspaceId: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (config.id === \"\") {\r\n result.result = false;\r\n result.error = \"configPost: invalid config ID\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create no parameter config endpoint\r\n let endpoint: string = mindlineConfig.configEndpoint();\r\n // create config headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // create config body\r\n let configBody: string = `\r\n {\r\n \"workspaceId\": \"${workspaceId}\",\r\n \"name\": \"${config.name}\",\r\n \"description\": \"${config.description}\",\r\n \"isEnabled\": ${config.isEnabled},\r\n \"tenants\": [`;\r\n config.tenants.map((tci) => {\r\n // be sure we send null and not \"null\" in body\r\n let sourceGroupId: string = tci.sourceGroupId != \"\" ? `\"${tci.sourceGroupId}\"` : \"null\";\r\n let sourceGroupName: string = tci.sourceGroupName != \"\" ? `\"${tci.sourceGroupName}\"` : \"null\";\r\n let targetGroupId: string = tci.targetGroupId != \"\" ? `\"${tci.targetGroupId}\"` : \"null\";\r\n let targetGroupName: string = tci.targetGroupName != \"\" ? `\"${tci.targetGroupName}\"` : \"null\";\r\n // if last character is } we need a comma first\r\n let needComma: boolean = configBody.slice(-1) === \"}\";\r\n if (needComma) configBody += \",\";\r\n configBody += `{\r\n \"tenantId\": \"${tci.tid}\",\r\n \"sourceGroupId\": ${sourceGroupId},\r\n \"sourceGroupName\": ${sourceGroupName},\r\n \"targetGroupId\": ${targetGroupId},\r\n \"targetGroupName\": ${targetGroupName},\r\n \"configurationTenantType\": \"${tci.configurationTenantType}\"\r\n }`;\r\n });\r\n configBody += `]}`;\r\n let options = { method: \"POST\", headers: headers, body: configBody };\r\n // make config endpoint call\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting POST to /config: \" + endpoint);\r\n let response = await fetch(endpoint, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let data = await response.json();\r\n config.id = data;\r\n console.log(\r\n `Successful ConfigID: ${data} from POST to /config: ${configBody}`\r\n );\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed PUT to /config: ${configBody}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.status = 500;\r\n result.result = false;\r\n result.error = error.message;\r\n console.log(result.error);\r\n return result;\r\n }\r\n}\r\n//configPut\r\nexport async function configPut(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n config: SyncConfig,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (config.id === \"\") {\r\n result.result = false;\r\n result.error = \"configPost: invalid config ID\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create parametrized config endpoint\r\n let endpoint: string = mindlineConfig.configEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"configurationId\", config.id);\r\n // create config headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // create config body\r\n let configBody: string = `\r\n {\r\n \"name\": \"${config.name}\",\r\n \"description\": \"${config.description}\",\r\n \"tenants\": [`;\r\n config.tenants.map((tci: TenantConfigInfo) => {\r\n // if last character is } we need a comma first\r\n let needComma: boolean = configBody.slice(-1) === \"}\";\r\n if (needComma) configBody += \",\";\r\n // be sure we send null and not \"null\" in body\r\n let sourceGroupId: string = tci.sourceGroupId != \"\" ? `\"${tci.sourceGroupId}\"` : \"null\";\r\n let sourceGroupName: string = tci.sourceGroupName != \"\" ? `\"${tci.sourceGroupName}\"` : \"null\";\r\n let targetGroupId: string = tci.targetGroupId != \"\" ? `\"${tci.targetGroupId}\"` : \"null\";\r\n let targetGroupName: string = tci.targetGroupName != \"\" ? `\"${tci.targetGroupName}\"` : \"null\";\r\n configBody += `{\r\n \"tenantId\": \"${tci.tid}\",\r\n \"sourceGroupId\": ${sourceGroupId},\r\n \"sourceGroupName\": ${sourceGroupName},\r\n \"targetGroupId\": ${targetGroupId},\r\n \"targetGroupName\": ${targetGroupName},\r\n \"configurationTenantType\": \"${tci.configurationTenantType}\",\r\n \"deltaToken\": \"${tci.deltaToken}\"\r\n }`;\r\n });\r\n configBody += `]}`;\r\n let options = { method: \"PUT\", headers: headers, body: configBody };\r\n // make config endpoint call\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting PUT to /config: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful PUT to ${url.href}: ${configBody}`);\r\n let data = await response.json();\r\n return {\r\n ...result,\r\n array: [data]\r\n };\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed PUT to ${url.href}: ${configBody}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//configsGet\r\nexport async function configsGet(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n workspaceID: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we need a workspace id\r\n if (workspaceID === \"\") {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"configsGet: no workspace provided\";\r\n return result;\r\n }\r\n // create endpoint\r\n let endpoint: string = mindlineConfig.configsEndpoint();\r\n // add parameter to endpoint\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"workspaceId\", workspaceID);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n // make endpoint call\r\n let options = { method: \"GET\", headers: headers };\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting GET from /configurations: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let returnedArray: Array<Object> = await response.json();\r\n if (returnedArray != null) {\r\n result.array = returnedArray;\r\n let initialValue: string = \"\";\r\n console.log(`Successful GET from /configurations: ${result.array.reduce((acc, curr: any) => acc + curr.name + \" \", initialValue)}`);\r\n return result;\r\n }\r\n else {\r\n result.error = `Failed GET from /configurations: failed to JSON-parse response`;\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n else {\r\n console.log(`Failed GET from /configurations: ${url.href}`);\r\n result.error = await processErrors(response);\r\n result.status = 500;\r\n result.result = false;\r\n console.log(result.error);\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//workspaceConfigsGET\r\nexport async function workspaceConfigsGet(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n workspaceId: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n\r\n if (!workspaceId) {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"workspaceConfigsGet: no workspace provided\";\r\n return result;\r\n }\r\n\r\n // Create endpoint\r\n let endpoint: string = mindlineConfig.workspaceConfigsEndpoint();\r\n // Add parameter to endpoint\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"workspaceId\", workspaceId);\r\n\r\n // Headers with Bearer token\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n\r\n let options = { method: \"GET\", headers };\r\n\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting GET from /workspaceConfigs: \" + url.href);\r\n\r\n const response = await fetch(url.href, options);\r\n\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n const returnedArray: Array<any> = await response.json();\r\n\r\n if (returnedArray) {\r\n // Filter by configurationType (sync or auditLog)\r\n result.array = returnedArray.map((config) => ({\r\n id: config.id,\r\n name: config.name,\r\n description: config.description,\r\n configurationType: config.configurationType,\r\n isEnabled: config.isEnabled,\r\n }));\r\n\r\n console.log(\r\n `workspaceConfigsGet: found ${result.array.length} configs.`\r\n );\r\n return result;\r\n } else {\r\n result.error =\r\n \"workspaceConfigsGet: failed to parse returned configs\";\r\n result.result = false;\r\n result.status = 500;\r\n }\r\n } else {\r\n result.error = await processErrors(response);\r\n result.result = false;\r\n result.status = response.status;\r\n console.log(result.error);\r\n }\r\n } catch (error: any) {\r\n result.error = error.message;\r\n result.result = false;\r\n result.status = 500;\r\n }\r\n\r\n return result;\r\n}\r\n//initPost\r\nexport async function initPost(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we expect valid company name and domain by this point\r\n if (user.companyName === \"\" || user.companyDomain === \"\") {\r\n result.result = false;\r\n result.error = \"initPost: invalid company name or domain\"\r\n result.status = 500;\r\n return result;\r\n }\r\n // create init endpoint\r\n let endpoint: string = mindlineConfig.initEndpoint();\r\n // create init headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n // create init body\r\n let initBody: string = `\r\n {\r\n \"tenantCreateModel\": {\r\n \"tenantId\": \"${user.tid}\",\r\n \"name\": \"${user.companyName}\",\r\n \"domain\": \"${user.companyDomain}\",\r\n \"type\": \"aad\",\r\n \"authority\": \"${user.authority}\"\r\n }\r\n }`;\r\n let options = { method: \"POST\", headers: headers, body: initBody };\r\n // make init endpoint call\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting POST to /configuration/init: \" + endpoint);\r\n let response = await fetch(endpoint, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful POST to /configuration/init: ${initBody}`);\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n result.status = 500;\r\n result.result = false;\r\n console.log(`Failed POST to /configuration/init: ${initBody}`);\r\n console.log(result.error);\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n console.log(result.error);\r\n }\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n}\r\n//tenantDelete\r\nexport async function tenantDelete(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n tenant: Tenant,\r\n workspaceId: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n debug = debug;\r\n let result: APIResult = new APIResult();\r\n // we expect valid tid amd workspaceId\r\n if (tenant.tid === \"\" || workspaceId === \"\") {\r\n result.result = false;\r\n result.error = \"tenantDelete: invalid tid, workspaceId\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create parametrized tenant endpoint\r\n let url: URL = new URL(mindlineConfig.tenantEndpoint());\r\n url.searchParams.append(\"tenantId\", tenant.tid);\r\n url.searchParams.append(\"workspaceId\", workspaceId);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // make tenant endpoint call\r\n let options = { method: \"DELETE\", headers: headers };\r\n try {\r\n console.log(\"Attempting DELETE from /tenant: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful DELETE from /tenant: ${url.href}`);\r\n return result;\r\n }\r\n else {\r\n console.log(`Failed DELETE from /tenant: ${url.href}`);\r\n result.error = await processErrors(response);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(result.error);\r\n }\r\n return result;\r\n}\r\n//tenantsGet\r\nexport async function tenantsGet(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n workspaceID: string,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we need a workspace id\r\n if (workspaceID === \"\") {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"tenantsGet: no workspace provided\";\r\n return result;\r\n }\r\n // create endpoint\r\n let endpoint: string = mindlineConfig.tenantsEndpoint();\r\n // add parameter to endpoint\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"workspaceId\", workspaceID);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n // make endpoint call\r\n let options = { method: \"GET\", headers: headers };\r\n try {\r\n if (debug) debugger;\r\n console.log(`Attempting GET from /tenants: ${url.href}`);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let returnedArray: Array<Object> = await response.json();\r\n if (returnedArray != null) {\r\n result.array = returnedArray;\r\n let initialValue: string = \"\";\r\n console.log(`Successful GET from /tenants: ${result.array.reduce((acc, curr: any) => acc + curr.domain + \" \", initialValue)}`);\r\n return result;\r\n }\r\n else {\r\n result.error = `Failed GET from /tenants: failed to JSON-parse response`;\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n else {\r\n console.log(`Failed GET from /tenants: ${url.href}`);\r\n result.error = await processErrors(response);\r\n result.status = 500;\r\n result.result = false;\r\n console.log(result.error);\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//tenantPost: write validated tenant to back end\r\nexport async function tenantPost(\r\n instance: IPublicClientApplication,\r\n addingUser: User,\r\n tenant: Tenant,\r\n workspaceId: string\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we expect valid tid, name, validated domain\r\n if (tenant.tid === \"\" || tenant.name === \"\" || tenant.domain === \"\") {\r\n result.result = false;\r\n result.error = \"tenantPost: invalid tid, name, domain\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create parametrized tenant endpoint\r\n let endpoint: string = mindlineConfig.tenantEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"workspaceId\", workspaceId);\r\n // create tenant headers\r\n const headers = await mindlineDefineHeaders(instance, addingUser);\r\n // create tenant body\r\n let tenantBody: string = `\r\n {\"tenantId\": \"${tenant.tid}\", \r\n \"name\": \"${tenant.name}\", \r\n \"domain\": \"${tenant.domain}\",\r\n \"type\": 1,\r\n \"authority\": \"${tenant.authority}\"\r\n }`;\r\n let options = { method: \"POST\", headers: headers, body: tenantBody };\r\n // make tenant endpoint call\r\n try {\r\n console.log(`Attempting POST to ${url.href}: ${tenantBody}`);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful POST to ${url.href}: ${tenantBody}`);\r\n return result;\r\n }\r\n else {\r\n console.log(`Failed POST to ${url.href}: ${tenantBody}`);\r\n result.error = await processErrors(response);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(result.error);\r\n }\r\n return result;\r\n}\r\n//workspacePut\r\nexport async function workspacePut(instance: IPublicClientApplication, authorizedUser: User, workspaceId: string, workspaceName: string): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (workspaceId == \"\" || workspaceName == \"\") {\r\n result.result = false;\r\n result.error = \"workspacePut: invalid workspace ID or name\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create parameterized workspace endpoint\r\n let endpoint: string = mindlineConfig.workspaceEndpoint();\r\n let url: URL = new URL(endpoint);\r\n url.searchParams.append(\"workspaceId\", workspaceId);\r\n url.searchParams.append(\"workspaceName\", workspaceName);\r\n // create workspace headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n let options = { method: \"PUT\", headers: headers };\r\n // make config endpoint call\r\n try {\r\n console.log(\"Attempting PUT to /workspace: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful PUT to ${url.href}`);\r\n return result;\r\n }\r\n else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed PUT to ${url.href}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//workspacesGet\r\nexport async function workspacesGet(\r\n instance: IPublicClientApplication,\r\n user: User,\r\n debug: boolean\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n // we need a valid email address\r\n if (user.mail == undefined || user.mail == \"\") {\r\n result.result = false;\r\n result.status = 500;\r\n result.error = \"adminsGet: no workspace provided\";\r\n return result;\r\n }\r\n // create workspaces endpoint\r\n let endpoint: string = mindlineConfig.workspacesEndpoint();\r\n // create workspace endpoint\r\n let url: URL = new URL(endpoint);\r\n // create workspace headers\r\n const headers = await mindlineDefineHeaders(instance, user);\r\n // make workspace endpoint call\r\n let options = { method: \"GET\", headers: headers };\r\n try {\r\n if (debug) debugger;\r\n console.log(\"Attempting GET from /workspaces endpoint: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n let returnedArray: Array<Object> = await response.json();\r\n if (returnedArray != null) {\r\n result.array = returnedArray;\r\n let initialValue: string = \"\";\r\n console.log(`Successful GET from /workspaces: ${result.array.reduce((acc, curr: any) => acc + curr.name + \" \", initialValue)}`);\r\n return result;\r\n }\r\n else {\r\n result.error = `Failed GET from /workspaces: failed to JSON-parse response`;\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n }\r\n else {\r\n console.log(`Failed GET from /workspaces: ${url.href}`);\r\n result.error = await processErrors(response);\r\n result.status = 500;\r\n result.result = false;\r\n console.log(result.error);\r\n return result;\r\n }\r\n }\r\n catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//readerPost\r\nexport async function readerPost(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n config: SyncConfig\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (instance == null || authorizedUser == null) {\r\n result.result = false;\r\n result.error = \"readerPost: invalid parameters\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create reader endpoint\r\n let readerEndpoint: string = mindlineConfig.readerStartSyncEndpoint();\r\n let url: URL = new URL(readerEndpoint);\r\n url.searchParams.append(\"configurationId\", config.id);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // make reader endpoint call\r\n let options = { method: \"POST\", headers: headers };\r\n try {\r\n console.log(\"Attempting POST to /startSync: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful POST to /startSync: ${readerEndpoint}`);\r\n let jsonResponse = await response.json();\r\n if (jsonResponse.PayloadStr != \"\") {\r\n result.array = JSON.parse(jsonResponse.PayloadStr);\r\n }\r\n else {\r\n result.result = false;\r\n result.error = \"readerPost: blank payload returned, sync may be disabled on back end\";\r\n result.status = 500;\r\n }\r\n return result;\r\n } else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed POST to /startSync: ${readerEndpoint}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n } catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}\r\n//readStats\r\nexport async function readerStats(\r\n instance: IPublicClientApplication,\r\n authorizedUser: User,\r\n tenantId: String\r\n): Promise<APIResult> {\r\n let result: APIResult = new APIResult();\r\n if (instance == null || authorizedUser == null) {\r\n result.result = false;\r\n result.error = \"readerPost: invalid parameters\";\r\n result.status = 500;\r\n return result;\r\n }\r\n // create reader endpoint\r\n let stasEndpoint: string = mindlineConfig.statsEndpoint();\r\n let url: URL = new URL(`${stasEndpoint}/${tenantId}`);\r\n // create headers\r\n const headers = await mindlineDefineHeaders(instance, authorizedUser);\r\n // make reader endpoint call\r\n let options = { method: \"GET\", headers: headers };\r\n try {\r\n console.log(\"Attempting GET to /stats: \" + url.href);\r\n let response = await fetch(url.href, options);\r\n if (response.status === 200 && response.statusText === \"OK\") {\r\n console.log(`Successful GET to /stats: ${stasEndpoint}`);\r\n let jsonResponse = await response.json();\r\n result.array = [jsonResponse];\r\n return result;\r\n } else if (response.status === 204) {\r\n result.error = await processErrors(response);\r\n console.log(`Failed GET to /stats: ${stasEndpoint}`);\r\n console.log(result.error);\r\n result.status = 204;\r\n result.result = false;\r\n return result;\r\n } else {\r\n result.error = await processErrors(response);\r\n console.log(`Failed GET to /stats: ${stasEndpoint}`);\r\n console.log(result.error);\r\n result.status = 500;\r\n result.result = false;\r\n return result;\r\n }\r\n } catch (error: any) {\r\n result.error = error.message;\r\n result.status = 500;\r\n result.result = false;\r\n console.log(error.message);\r\n }\r\n return result;\r\n}"],"names":["FILTER_FIELD","sum","a","b","helloNpm","getSyncVersion","APIResult","__publicField","azureConfig","_mindlineConfig","mindlineConfig","graphConfig","Group","UserScope","User","TenantType","TenantPermissionType","Tenant","getAppId","authority","tct","getGraphEndpoint","getLoginEndpoint","TenantConfigType","TenantConfigInfo","SyncConfig","AuditConfig","Workspace","storageAvailable","storage","x","e","InitInfo","bClearLocalStorage","__privateAdd","_InitInfo_instances","result","iiReadFromLocalStorage","__privateMethod","initFromObjects_fn","usersString","users","tenantsString","tenants","configsString","configs","workspacesString","workspaces","deserializeArray","initInfoString","item","workspace","userID","user","currentUser","tenantID","tenant","currentTenant","configID","config","currentConfig","ii","newuser","newtenant","newconfig","newworkspace","tasksData","TaskArray","_TaskArray_instances","Task","taskArray","initTasksFromObjects_fn","taskType","startDate","task","findTask_fn","save_fn","endDate","status","tasks","t","newTask","st","newSubtask","taskArrayString","minuteAdjustment","elapsedSeconds","elapsedMinutes","Milestone","run","start","post","read","write","MilestoneArray","_MilestoneArray_instances","milestones","syncmilestones","milestonesString","setMilestones","currentRun","newMilestone","ms","newms","removedMilestone","removed","milestone","BatchArray","syncPortalGlobalState","tciPotentialSource","sourceTenant","sourceTenantNode","TenantNode","tciPotentialTarget","targetTenant","targetTenantNode","setSyncProgress","setConfigSyncResult","setIdleText","tn","ta","prev","next","batchIdArray","setRefreshDeltaTrigger","setReadersTotal","setReadersExcluded","setReadersCurrent","setWritersTotal","setWritersExcluded","setWritersCurrent","message","handler","connection","matchingPair","o","tenantNode","statsarray","statskeys","statsvalues","bTotalCountZero","bCurrentCountZero","j","bTotalCount","bCurrentCount","bExcludedCount","bDeferredCount","bRescheduledCount","writerNode","tidRegexp","matchTID","bReadingComplete","bWritingComplete","bWritingStarted","bNothingToSync","readerTotal","readerCurrent","readerExcluded","writerTotal","writerCurrent","writerExcluded","expectedPercentPerSecond","batchPair","endpoint","endpointUrl","signalR","error","connectionId","instance","authorizedUser","readerPost","tid","name","batchId","total","excluded","written","deferred","ResourceArray","bInitialize","resourceArray","resourceNodesString","resources","ResourceNode","readResources","raString","type","resource","cost","ActorArray","aaFromLocalStorage","actorsString","actors","ActorNode","actor","role","updatedby","updatedon","getGraphAPIScope","graphDefineHeaders","headers","graphAPIScope","response","accounts","homeAccountId","account","groupsGet","groupSearchString","options","groupsEndpoint","data","oauth2PermissionGrantsGet","spid","oid","spurl","url","oauth2PermissionGrantsSet","loggedInUser","id","scopes","grantsurl","scopesBody","requestAdminConsent","admin","adminConsentURL","clientId","servicePrincipalGet","appid","signIn","tenantUnauthenticatedLookup","signinURL","regex","regexMatch","domain","signInIncrementally","scope","challengeURL","signOut","userEndpoint","userLogoutResponse","signoutURL","tenantRelationshipsGetByDomain","debug","bearer","tenantEndpoint","tenantRelationshipsGetById","endpoints","regexes","openidEndpoint","tenantAuthEndpoint","k","authMatches","userDelegatedScopesGet","grants","userDelegatedScopesRemove","usersGet","usersEndpoint","auditConfigAdd","ac","auditConfigPost","auditConfigRetrieve","configurationId","auditConfigGet","auditConfigEdit","auditConfigPut","auditConfigRemove","auditConfigDelete","auditEventsRetrieve","auditEventsGet","configEdit","setConfigId","setSelectedConfigs","configPost","newSelection","idx","configPut","configEnable","enabled","configPatch","configConsentForRead","configId","consent","configConsentReadPut","configConsentForWrite","configConsentWritePut","configsRefresh","workspaceId","w","configGet","configRemove","configDelete","initGet","initPost","workspaceInfoGet","tenantAdd","tenantPost","tenantRemove","tenantDelete","userAdd","adminPost","userRemove","adminDelete","workspaceEdit","workspaceName","workspacePut","processReturnedAdmins","returnedAdmins","adminSelectedId","usIndex","u","dummyIndex","processReturnedTenants","returnedTenants","tenantSelectedId","tsIndex","processReturnedConfigs","returnedConfigs","configSelectedId","csIndex","c","tci","tenantConfigInfo","workspacesGet","wsIndex","cid","adminsPromise","adminsGet","tenantsPromise","tenantsGet","configsPromise","configsGet","adminsResult","tenantsResult","configsResult","getPowerBIAccessToken","accesstoken","syncversion","azureDefineHeaders","i","canListRootAssignments","listrootassignmentsEndpoint","processErrors","elevateGlobalAdminToUserAccessAdmin","elevateaccessEndpoint","getAPIScope","apiAppID","mindlineDefineHeaders","apiScope","errorString","acc","curr","workspaceID","returnedArray","adminBody","configBody","json","configConsentReadBody","configConsentWriteBody","sourceGroupId","sourceGroupName","targetGroupId","targetGroupName","workspaceConfigsGet","initBody","addingUser","tenantBody","readerEndpoint","jsonResponse","readerStats","tenantId","stasEndpoint"],"mappings":"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"}
|