From 1db79bafffd64e73c5c462d263792e112712d3a1 Mon Sep 17 00:00:00 2001 From: Ryan Slatten Date: Thu, 31 Oct 2024 10:57:27 -0400 Subject: [PATCH] updates from pr comments --- .../src/ArcGISIdentityManagerFactory.ts | 15 +++++---- .../service/src/ObservationProcessor.ts | 32 +++++++++++++++++-- plugins/arcgis/service/src/index.ts | 15 ++++----- .../projects/main/src/lib/ArcGISConfig.ts | 10 ------ 4 files changed, 43 insertions(+), 29 deletions(-) diff --git a/plugins/arcgis/service/src/ArcGISIdentityManagerFactory.ts b/plugins/arcgis/service/src/ArcGISIdentityManagerFactory.ts index 586610df7..dafadec96 100644 --- a/plugins/arcgis/service/src/ArcGISIdentityManagerFactory.ts +++ b/plugins/arcgis/service/src/ArcGISIdentityManagerFactory.ts @@ -1,14 +1,13 @@ -import { ArcGISIdentityManager } from "@esri/arcgis-rest-request" +import { ArcGISIdentityManager, request } from "@esri/arcgis-rest-request" import { ArcGISAuthConfig, AuthType, FeatureServiceConfig, OAuthAuthConfig, TokenAuthConfig, UsernamePasswordAuthConfig } from './ArcGISConfig' -import { HttpClient } from "./HttpClient"; import { ObservationProcessor } from "./ObservationProcessor"; interface ArcGISIdentityManagerFactory { - create(portal: string, server: string, config: ArcGISAuthConfig, httpClient?: HttpClient, processor?: ObservationProcessor): Promise + create(portal: string, server: string, config: ArcGISAuthConfig, processor?: ObservationProcessor): Promise } const OAuthIdentityManagerFactory: ArcGISIdentityManagerFactory = { - async create(portal: string, server: string, auth: OAuthAuthConfig, httpClient: HttpClient, processor: ObservationProcessor): Promise { + async create(portal: string, server: string, auth: OAuthAuthConfig, processor: ObservationProcessor): Promise { console.debug('Client ID provided for authentication') const { clientId, authToken, authTokenExpires, refreshToken, refreshTokenExpires } = auth @@ -24,7 +23,10 @@ const OAuthIdentityManagerFactory: ArcGISIdentityManagerFactory = { // TODO: find a way without using constructor nor httpClient const url = `${portal}/oauth2/token?client_id=${clientId}&refresh_token=${refreshToken}&grant_type=refresh_token` try { - const response = await httpClient.sendGet(url) + const response = await request(url, { + httpMethod: 'GET' + }); + // Update authToken to new token const config = await processor.safeGetConfig(); let service = config.featureServices.find(service => service.url === portal)?.auth as OAuthAuthConfig; @@ -85,7 +87,6 @@ const authConfigMap: { [key: string]: ArcGISIdentityManagerFactory } = { export function getIdentityManager( config: FeatureServiceConfig, - httpClient: HttpClient, // TODO remove in favor of an open source lib like axios processor: ObservationProcessor ): Promise { const auth = config.auth @@ -97,7 +98,7 @@ export function getIdentityManager( if (!factory) { throw new Error(`No factory found for type ${authType}`) } - return factory.create(getPortalUrl(config.url), getServerUrl(config.url), auth, httpClient, processor) + return factory.create(getPortalUrl(config.url), getServerUrl(config.url), auth, processor) } diff --git a/plugins/arcgis/service/src/ObservationProcessor.ts b/plugins/arcgis/service/src/ObservationProcessor.ts index 7bf5ab172..a5b9a6e94 100644 --- a/plugins/arcgis/service/src/ObservationProcessor.ts +++ b/plugins/arcgis/service/src/ObservationProcessor.ts @@ -14,7 +14,7 @@ import { EventTransform } from './EventTransform'; import { GeometryChangedHandler } from './GeometryChangedHandler'; import { EventDeletionHandler } from './EventDeletionHandler'; import { EventLayerProcessorOrganizer } from './EventLayerProcessorOrganizer'; -import { FeatureServiceConfig, FeatureLayerConfig, AuthType } from "./ArcGISConfig" +import { FeatureServiceConfig, FeatureLayerConfig, AuthType, OAuthAuthConfig } from "./ArcGISConfig" import { PluginStateRepository } from '@ngageoint/mage.service/lib/plugins.api' import { FeatureServiceAdmin } from './FeatureServiceAdmin'; @@ -120,8 +120,13 @@ export class ObservationProcessor { * Gets the current configuration from the database. * @returns The current configuration from the database. */ - public async safeGetConfig(): Promise { - return await this._stateRepo.get().then(x => !!x ? x : this._stateRepo.put(defaultArcGISPluginConfig)) + public async safeGetConfig(showFeatureAuth?: boolean): Promise { + const state = await this._stateRepo.get(); + if (!state) return await this._stateRepo.put(defaultArcGISPluginConfig); + if (!showFeatureAuth) { + state.featureServices = state.featureServices.map((service) => this.sanitizeFeatureService(service, AuthType.OAuth)); + } + return state; } /** @@ -132,6 +137,14 @@ export class ObservationProcessor { return await this._stateRepo.put(newConfig); } + /** + * Updates the confguration in the state repo. + * @param newConfig The new config to put into the state repo. + */ + public async patchConfig(newConfig: ArcGISPluginConfig): Promise { + return await this._stateRepo.patch(newConfig); + } + /** * Gets the current configuration and updates the processor if needed * @returns The current configuration from the database. @@ -151,6 +164,19 @@ export class ObservationProcessor { return config } + private sanitizeFeatureService(config: FeatureServiceConfig, type: AuthType): FeatureServiceConfig { + if (type === AuthType.OAuth) { + const newAuth = Object.assign({}, config.auth) as OAuthAuthConfig; + delete newAuth.refreshToken; + delete newAuth.refreshTokenExpires; + return { + ...config, + auth: newAuth + } + } + return config; + } + /** * Starts the processor. */ diff --git a/plugins/arcgis/service/src/index.ts b/plugins/arcgis/service/src/index.ts index ad93a4716..206603485 100644 --- a/plugins/arcgis/service/src/index.ts +++ b/plugins/arcgis/service/src/index.ts @@ -7,7 +7,6 @@ import { SettingPermission } from '@ngageoint/mage.service/lib/entities/authoriz import { ArcGISPluginConfig } from './ArcGISPluginConfig' import { AuthType } from './ArcGISConfig' import { ObservationProcessor } from './ObservationProcessor' -import { HttpClient } from './HttpClient' import { ArcGISIdentityManager, request } from "@esri/arcgis-rest-request" import { FeatureServiceConfig } from './ArcGISConfig' import { URL } from "node:url" @@ -157,7 +156,7 @@ const arcgisPluginHooks: InitPluginHook = { console.info('Applying ArcGIS plugin config...') const arcConfig = req.body as ArcGISPluginConfig const configString = JSON.stringify(arcConfig) - processor.putConfig(arcConfig) + processor.patchConfig(arcConfig) res.sendStatus(200) }) @@ -179,17 +178,16 @@ const arcgisPluginHooks: InitPluginHook = { } try { - const httpClient = new HttpClient(console) // Create the IdentityManager instance to validate credentials - await getIdentityManager(service!, httpClient, processor) + await getIdentityManager(service!, processor) let existingService = config.featureServices.find(service => service.url === url) if (existingService) { existingService = { ...existingService } } else { config.featureServices.push(service) } - - await processor.putConfig(config) + console.log('values patch') + await processor.patchConfig(config) return res.send(service) } catch (err) { return res.send('Invalid credentials provided to communicate with feature service').status(400) @@ -203,10 +201,9 @@ const arcgisPluginHooks: InitPluginHook = { if (!featureService) { return res.status(400) } - - const httpClient = new HttpClient(console) + try { - const identityManager = await getIdentityManager(featureService, httpClient, processor) + const identityManager = await getIdentityManager(featureService, processor) const response = await request(url, { authentication: identityManager }) diff --git a/plugins/arcgis/web-app/projects/main/src/lib/ArcGISConfig.ts b/plugins/arcgis/web-app/projects/main/src/lib/ArcGISConfig.ts index 91ff97475..c59cb6b91 100644 --- a/plugins/arcgis/web-app/projects/main/src/lib/ArcGISConfig.ts +++ b/plugins/arcgis/web-app/projects/main/src/lib/ArcGISConfig.ts @@ -126,16 +126,6 @@ export interface OAuthAuthConfig { * The expiration date for the temporary token */ authTokenExpires?: string - - /** - * The Refresh token for OAuth - */ - refreshToken?: string - - /** - * The expiration date for the Refresh token - */ - refreshTokenExpires?: string } /**