From 07294e6c0d3058d8cd6ee551077afa8d2ac95b8f Mon Sep 17 00:00:00 2001 From: Suchit Sahoo Date: Mon, 6 May 2024 16:55:04 +0000 Subject: [PATCH] Add Server Side batching for UI Metric Collectors Signed-off-by: Suchit Sahoo --- .../saved_objects/service/lib/repository.ts | 13 ++- .../usage_collection/common/constants.ts | 1 + src/plugins/usage_collection/server/config.ts | 8 +- src/plugins/usage_collection/server/plugin.ts | 1 + .../server/report/store_report.test.ts | 4 +- .../server/report/store_report.ts | 10 +- .../usage_collection/server/routes/index.ts | 3 +- .../server/routes/report_metrics.ts | 98 ++++++++++++++++++- src/plugins/usage_collection/server/types.ts | 34 +++++++ 9 files changed, 155 insertions(+), 17 deletions(-) create mode 100644 src/plugins/usage_collection/server/types.ts diff --git a/src/core/server/saved_objects/service/lib/repository.ts b/src/core/server/saved_objects/service/lib/repository.ts index 1f00976653cf..b7282ecdaf80 100644 --- a/src/core/server/saved_objects/service/lib/repository.ts +++ b/src/core/server/saved_objects/service/lib/repository.ts @@ -1542,12 +1542,13 @@ export class SavedObjectsRepository { } /** - * Increases a counter field by one. Creates the document if one doesn't exist for the given id. + * Increases a counter field by incrementValue which by default is 1. Creates the document if one doesn't exist for the given id. * * @param {string} type * @param {string} id * @param {string} counterFieldName * @param {object} [options={}] + * @param {number} [incrementValue=1] * @property {object} [options.migrationVersion=undefined] * @returns {promise} */ @@ -1555,7 +1556,8 @@ export class SavedObjectsRepository { type: string, id: string, counterFieldName: string, - options: SavedObjectsIncrementCounterOptions = {} + options: SavedObjectsIncrementCounterOptions = {}, + incrementValue: number = 1 ): Promise { if (typeof type !== 'string') { throw new Error('"type" argument must be a string'); @@ -1579,19 +1581,17 @@ export class SavedObjectsRepository { } else if (this._registry.isMultiNamespace(type)) { savedObjectNamespaces = await this.preflightGetNamespaces(type, id, namespace); } - const migrated = this._migrator.migrateDocument({ id, type, ...(savedObjectNamespace && { namespace: savedObjectNamespace }), ...(savedObjectNamespaces && { namespaces: savedObjectNamespaces }), - attributes: { [counterFieldName]: 1 }, + attributes: { [counterFieldName]: incrementValue }, migrationVersion, updated_at: time, }); const raw = this._serializer.savedObjectToRaw(migrated as SavedObjectSanitizedDoc); - const { body } = await this.client.update({ id: raw._id, index: this.getIndexForType(type), @@ -1610,7 +1610,7 @@ export class SavedObjectsRepository { `, lang: 'painless', params: { - count: 1, + count: incrementValue, time, type, counterFieldName, @@ -1619,7 +1619,6 @@ export class SavedObjectsRepository { upsert: raw._source, }, }); - const { originId } = body.get?._source ?? {}; return { id, diff --git a/src/plugins/usage_collection/common/constants.ts b/src/plugins/usage_collection/common/constants.ts index c83b2c38a27f..1417666a02b9 100644 --- a/src/plugins/usage_collection/common/constants.ts +++ b/src/plugins/usage_collection/common/constants.ts @@ -30,3 +30,4 @@ export const OPENSEARCH_DASHBOARDS_STATS_TYPE = 'opensearch_dashboards_stats'; export const DEFAULT_MAXIMUM_WAIT_TIME_FOR_ALL_COLLECTORS_IN_S = 60; +export const DEFAULT_BATCHING_INTERVAL_FOR_UI_METRIC_IN_S = 60; diff --git a/src/plugins/usage_collection/server/config.ts b/src/plugins/usage_collection/server/config.ts index 5bc9ec500cdb..12b9f3aa3cd8 100644 --- a/src/plugins/usage_collection/server/config.ts +++ b/src/plugins/usage_collection/server/config.ts @@ -30,12 +30,18 @@ import { schema, TypeOf } from '@osd/config-schema'; import { PluginConfigDescriptor } from 'opensearch-dashboards/server'; -import { DEFAULT_MAXIMUM_WAIT_TIME_FOR_ALL_COLLECTORS_IN_S } from '../common/constants'; +import { + DEFAULT_BATCHING_INTERVAL_FOR_UI_METRIC_IN_S, + DEFAULT_MAXIMUM_WAIT_TIME_FOR_ALL_COLLECTORS_IN_S, +} from '../common/constants'; export const configSchema = schema.object({ uiMetric: schema.object({ enabled: schema.boolean({ defaultValue: false }), debug: schema.boolean({ defaultValue: schema.contextRef('dev') }), + batchingInterval: schema.number({ + defaultValue: DEFAULT_BATCHING_INTERVAL_FOR_UI_METRIC_IN_S, + }), }), maximumWaitTimeForAllCollectorsInS: schema.number({ defaultValue: DEFAULT_MAXIMUM_WAIT_TIME_FOR_ALL_COLLECTORS_IN_S, diff --git a/src/plugins/usage_collection/server/plugin.ts b/src/plugins/usage_collection/server/plugin.ts index 92e6e31bd7d8..fab8d4dbe5dd 100644 --- a/src/plugins/usage_collection/server/plugin.ts +++ b/src/plugins/usage_collection/server/plugin.ts @@ -75,6 +75,7 @@ export class UsageCollectionPlugin implements Plugin { opensearchDashboardsVersion: this.initializerContext.env.packageInfo.version, server: core.http.getServerInfo(), uuid: this.initializerContext.env.instanceUuid, + batchingInterval: config.uiMetric.batchingInterval, }, metrics: core.metrics, overallStatus$: core.status.overall$, diff --git a/src/plugins/usage_collection/server/report/store_report.test.ts b/src/plugins/usage_collection/server/report/store_report.test.ts index 774fb5ac1c6e..97891674be18 100644 --- a/src/plugins/usage_collection/server/report/store_report.test.ts +++ b/src/plugins/usage_collection/server/report/store_report.test.ts @@ -80,7 +80,9 @@ describe('store_report', () => { expect(savedObjectClient.incrementCounter).toHaveBeenCalledWith( 'ui-metric', 'test-app-name:test-event-name', - 'count' + 'count', + {}, + 3 ); expect(savedObjectClient.bulkCreate).toHaveBeenCalledWith([ { diff --git a/src/plugins/usage_collection/server/report/store_report.ts b/src/plugins/usage_collection/server/report/store_report.ts index d9aad5132d0c..25acc114dbe3 100644 --- a/src/plugins/usage_collection/server/report/store_report.ts +++ b/src/plugins/usage_collection/server/report/store_report.ts @@ -57,11 +57,17 @@ export async function storeReport( }; }), ...uiStatsMetrics.map(async ([key, metric]) => { - const { appName, eventName } = metric; + const { appName, eventName, stats } = metric; const savedObjectId = `${appName}:${eventName}`; return { saved_objects: [ - await internalRepository.incrementCounter('ui-metric', savedObjectId, 'count'), + await internalRepository.incrementCounter( + 'ui-metric', + savedObjectId, + 'count', + {}, + stats.sum + ), ], }; }), diff --git a/src/plugins/usage_collection/server/routes/index.ts b/src/plugins/usage_collection/server/routes/index.ts index 2b83058afbcc..a18200fcb8a4 100644 --- a/src/plugins/usage_collection/server/routes/index.ts +++ b/src/plugins/usage_collection/server/routes/index.ts @@ -56,11 +56,12 @@ export function setupRoutes({ hostname: string; port: number; }; + batchingInterval: number; }; collectorSet: CollectorSet; metrics: MetricsServiceSetup; overallStatus$: Observable; }) { - registerUiMetricRoute(router, getSavedObjects); + registerUiMetricRoute(router, getSavedObjects, rest.config.batchingInterval); registerStatsRoute({ router, ...rest }); } diff --git a/src/plugins/usage_collection/server/routes/report_metrics.ts b/src/plugins/usage_collection/server/routes/report_metrics.ts index 1964e1bf27e1..e9e60a84084b 100644 --- a/src/plugins/usage_collection/server/routes/report_metrics.ts +++ b/src/plugins/usage_collection/server/routes/report_metrics.ts @@ -31,11 +31,16 @@ import { schema } from '@osd/config-schema'; import { IRouter, ISavedObjectsRepository } from 'opensearch-dashboards/server'; import { storeReport, reportSchema } from '../report'; +import { BatchReport } from '../types'; +import { ReportSchemaType } from '../report/schema'; export function registerUiMetricRoute( router: IRouter, - getSavedObjects: () => ISavedObjectsRepository | undefined + getSavedObjects: () => ISavedObjectsRepository | undefined, + batchingInterval: number ) { + let batchReport = { report: {}, startTimeStamp: 0 } as BatchReport; + const bactchingIntervalInMs = batchingInterval * 1000; router.post( { path: '/api/ui_metric/report', @@ -48,11 +53,30 @@ export function registerUiMetricRoute( async (context, req, res) => { const { report } = req.body; try { - const internalRepository = getSavedObjects(); - if (!internalRepository) { - throw Error(`The saved objects client hasn't been initialised yet`); + const currDate = new Date(); + const currTime = currDate.getTime(); + + // Add the current report to batchReport + batchReport.report = combineReports(report, batchReport.report); + // If the time duration since the batchReport startTime is greater than batchInterval then write it to the savedObject + if (currTime - batchReport.startTimeStamp >= bactchingIntervalInMs) { + const prevReport = batchReport; + + batchReport = { + report: {}, + startTimeStamp: currTime, + }; // reseting the batchReport and updating the startTimestamp to current TimeStamp + + if (prevReport) { + // Write the previously batched Report to the saved object + const internalRepository = getSavedObjects(); + if (!internalRepository) { + throw Error(`The saved objects client hasn't been initialised yet`); + } + await storeReport(internalRepository, prevReport.report); + } } - await storeReport(internalRepository, report); + return res.ok({ body: { status: 'ok' } }); } catch (error) { return res.ok({ body: { status: 'fail' } }); @@ -60,3 +84,67 @@ export function registerUiMetricRoute( } ); } + +function combineReports(report1: ReportSchemaType, report2: ReportSchemaType) { + // Combines report2 onto the report1 and returns the updated report1 + + // Combining User Agents + const combinedUserAgent = { ...report1.userAgent }; + + if (report2.userAgent !== undefined) { + for (const key in Object.keys(report2.userAgent)) { + if (report2.userAgent[key] === undefined) { + continue; + } + + // Only adding the useragents not already present in report1 + if (!report1.userAgent || report1.userAgent[key] === undefined) { + combinedUserAgent[key] = report2.userAgent[key]; + } + } + } + + // Combining UI metrics + const combinedUIMetric = { ...report1.uiStatsMetrics }; + if (report2.uiStatsMetrics !== undefined) { + for (const key of Object.keys(report2.uiStatsMetrics)) { + if (report2.uiStatsMetrics[key] === undefined) { + continue; + } else if ( + report1.uiStatsMetrics === undefined || + report1.uiStatsMetrics[key] === undefined + ) { + combinedUIMetric[key] = report2.uiStatsMetrics[key]; + } else { + const { stats, ...rest } = combinedUIMetric[key]; + const combinedStats = { ...stats }; + combinedStats.sum += report2.uiStatsMetrics[key].stats.sum; // Updating the sum since it is field we will be using to update the saved Object + combinedUIMetric[key] = { ...rest, stats: combinedStats }; + } + } + } + + // Combining Application Usage + const combinedApplicationUsage = { ...report1.application_usage }; + if (report2.application_usage !== undefined) { + for (const key in Object.keys(report2.application_usage)) { + if (report2.application_usage[key] === undefined) { + continue; + } else if (!report1.application_usage || report1.application_usage[key] === undefined) { + combinedApplicationUsage[key] = report2.application_usage[key]; + } else { + const combinedUsage = { ...combinedApplicationUsage[key] }; + combinedUsage.numberOfClicks += report2.application_usage[key].numberOfClicks; + combinedUsage.minutesOnScreen += report2.application_usage[key].minutesOnScreen; + combinedApplicationUsage[key] = combinedUsage; + } + } + } + + return { + reportVersion: report1.reportVersion, + userAgent: combinedUserAgent, + uiStatsMetrics: combinedUIMetric, + application_usage: combinedApplicationUsage, + } as ReportSchemaType; +} diff --git a/src/plugins/usage_collection/server/types.ts b/src/plugins/usage_collection/server/types.ts new file mode 100644 index 000000000000..ce7d1ec02315 --- /dev/null +++ b/src/plugins/usage_collection/server/types.ts @@ -0,0 +1,34 @@ +/* + * SPDX-License-Identifier: Apache-2.0 + * + * The OpenSearch Contributors require contributions made to + * this file be licensed under the Apache-2.0 license or a + * compatible open source license. + * + * Any modifications Copyright OpenSearch Contributors. See + * GitHub history for details. + */ + +/* + * Licensed to Elasticsearch B.V. under one or more contributor + * license agreements. See the NOTICE file distributed with + * this work for additional information regarding copyright + * ownership. Elasticsearch B.V. licenses this file to you under + * the Apache License, Version 2.0 (the "License"); you may + * not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + */ +import { ReportSchemaType } from './report/schema'; +export interface BatchReport { + report: ReportSchemaType; + startTimeStamp: number; +}