Skip to content

Commit

Permalink
Add Server Side batching for UI Metric Collectors
Browse files Browse the repository at this point in the history
Signed-off-by: Suchit Sahoo <[email protected]>
  • Loading branch information
LDrago27 committed May 6, 2024
1 parent 5258d83 commit b661d78
Show file tree
Hide file tree
Showing 9 changed files with 157 additions and 17 deletions.
13 changes: 6 additions & 7 deletions src/core/server/saved_objects/service/lib/repository.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1542,20 +1542,22 @@ 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}
*/
async incrementCounter(
type: string,
id: string,
counterFieldName: string,
options: SavedObjectsIncrementCounterOptions = {}
options: SavedObjectsIncrementCounterOptions = {},
incrementValue: number = 1
): Promise<SavedObject> {
if (typeof type !== 'string') {
throw new Error('"type" argument must be a string');
Expand All @@ -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<SavedObjectsRawDocSource>({
id: raw._id,
index: this.getIndexForType(type),
Expand All @@ -1610,7 +1610,7 @@ export class SavedObjectsRepository {
`,
lang: 'painless',
params: {
count: 1,
count: incrementValue,
time,
type,
counterFieldName,
Expand All @@ -1619,7 +1619,6 @@ export class SavedObjectsRepository {
upsert: raw._source,
},
});

const { originId } = body.get?._source ?? {};
return {
id,
Expand Down
1 change: 1 addition & 0 deletions src/plugins/usage_collection/common/constants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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;
8 changes: 7 additions & 1 deletion src/plugins/usage_collection/server/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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,
Expand Down
1 change: 1 addition & 0 deletions src/plugins/usage_collection/server/plugin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,7 @@ export class UsageCollectionPlugin implements Plugin<CollectorSet> {
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$,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -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([
{
Expand Down
10 changes: 8 additions & 2 deletions src/plugins/usage_collection/server/report/store_report.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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
),
],
};
}),
Expand Down
3 changes: 2 additions & 1 deletion src/plugins/usage_collection/server/routes/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -56,11 +56,12 @@ export function setupRoutes({
hostname: string;
port: number;
};
batchingInterval: number;
};
collectorSet: CollectorSet;
metrics: MetricsServiceSetup;
overallStatus$: Observable<ServiceStatus>;
}) {
registerUiMetricRoute(router, getSavedObjects);
registerUiMetricRoute(router, getSavedObjects, rest.config.batchingInterval);
registerStatsRoute({ router, ...rest });
}
100 changes: 95 additions & 5 deletions src/plugins/usage_collection/server/routes/report_metrics.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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',
Expand All @@ -48,15 +53,100 @@ 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' } });
}
}
);
}

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) {
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) {
for (const key in Object.keys(report2.uiStatsMetrics)) {
if (report2.uiStatsMetrics[key] === undefined) {
continue;
}
if (!report1.uiStatsMetrics || 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) {
for (const key in Object.keys(report2.application_usage)) {
if (report2.application_usage[key] === undefined) {
continue;
}

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;
}
34 changes: 34 additions & 0 deletions src/plugins/usage_collection/server/types.ts
Original file line number Diff line number Diff line change
@@ -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;
}

0 comments on commit b661d78

Please sign in to comment.