Skip to content

Commit 31aff2f

Browse files
LaVLaSmaroromanandrewballantyne
authored
Admin logging (opendatahub-io#914)
* Add admin activity logging Signed-off-by: Landon LaSmith <[email protected]> * Reduce risk of perf issues with added logging --------- Signed-off-by: Landon LaSmith <[email protected]> Co-authored-by: Maros Roman <[email protected]> Co-authored-by: Andrew Ballantyne <[email protected]>
1 parent 36e0af6 commit 31aff2f

File tree

9 files changed

+88
-2
lines changed

9 files changed

+88
-2
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88
.env*.local
99

1010
node_modules
11+
logs
1112

1213
.docs
1314
.gitkeep

backend/src/app.ts

+7
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,17 @@
1+
import * as fs from 'fs';
12
import * as path from 'path';
3+
import { LOG_DIR } from './utils/constants';
24
import fastifyStatic from 'fastify-static';
35
import fastifyAutoload from 'fastify-autoload';
46
import fastifySensible from 'fastify-sensible';
57
import { FastifyInstance } from 'fastify/types/instance';
68

79
export const initializeApp = async (fastify: FastifyInstance, opts: any): Promise<void> => {
10+
if (!fs.existsSync(LOG_DIR)) {
11+
fastify.log.info(`${LOG_DIR} does not exist. Creating`);
12+
fs.mkdirSync(LOG_DIR);
13+
}
14+
815
fastify.register(fastifySensible);
916

1017
fastify.register(fastifyStatic, {

backend/src/routes/api/gpu/index.ts

+5-2
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,11 @@
1-
import { KubeFastifyInstance } from '../../../types';
1+
import { KubeFastifyInstance, OauthFastifyRequest } from '../../../types';
22
import { getGPUNumber } from './gpuUtils';
3+
import { logRequestDetails } from '../../../utils/fileUtils';
34

45
export default async (fastify: KubeFastifyInstance): Promise<void> => {
5-
fastify.get('/', async () => {
6+
fastify.get('/', async (request: OauthFastifyRequest) => {
7+
logRequestDetails(fastify, request);
8+
69
return getGPUNumber(fastify);
710
});
811
};

backend/src/routes/api/k8s/index.ts

+3
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ import { FastifyReply, FastifyRequest } from 'fastify';
22
import { KubeFastifyInstance } from '../../../types';
33
import { USER_ACCESS_TOKEN } from '../../../utils/constants';
44
import { passThrough } from './pass-through';
5+
import { logRequestDetails } from '../../../utils/fileUtils';
56

67
module.exports = async (fastify: KubeFastifyInstance) => {
78
const kc = fastify.kube.config;
@@ -23,6 +24,8 @@ module.exports = async (fastify: KubeFastifyInstance) => {
2324
}>,
2425
reply: FastifyReply,
2526
) => {
27+
logRequestDetails(fastify, req);
28+
2629
const data = JSON.stringify(req.body);
2730
const kubeUri = req.params['*'];
2831
let url = `${cluster.server}/${kubeUri}`;

backend/src/routes/api/namespaces/index.ts

+3
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,14 @@
11
import { KubeFastifyInstance, OauthFastifyRequest } from '../../../types';
22
import { applyNamespaceChange } from './namespaceUtils';
33
import { NamespaceApplicationCase } from './const';
4+
import { logRequestDetails } from '../../../utils/fileUtils';
45

56
export default async (fastify: KubeFastifyInstance): Promise<void> => {
67
fastify.get(
78
'/:name/:context',
89
async (request: OauthFastifyRequest<{ Params: { name: string; context: string } }>) => {
10+
logRequestDetails(fastify, request);
11+
912
const { name, context: contextAsString } = request.params;
1013

1114
const context = parseInt(contextAsString) as NamespaceApplicationCase;

backend/src/routes/api/prometheus/index.ts

+3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import { KubeFastifyInstance, OauthFastifyRequest, PrometheusResponse } from '../../../types';
22
import { callPrometheus } from '../../../utils/prometheusUtils';
33
import { createCustomError } from '../../../utils/requestUtils';
4+
import { logRequestDetails } from '../../../utils/fileUtils';
45

56
module.exports = async (fastify: KubeFastifyInstance) => {
67
/**
@@ -12,6 +13,8 @@ module.exports = async (fastify: KubeFastifyInstance) => {
1213
async (
1314
request: OauthFastifyRequest<{ Body: { query: string; namespace: string } }>,
1415
): Promise<{ code: number; response: PrometheusResponse }> => {
16+
logRequestDetails(fastify, request);
17+
1518
const { query, namespace } = request.body;
1619

1720
return callPrometheus(fastify, request, query, namespace).catch((e) => {

backend/src/utils/constants.ts

+2
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,11 @@
1+
import * as path from 'path';
12
import './dotenv';
23
import { DashboardConfig, NotebookSize } from '../types';
34

45
export const PORT = process.env.PORT || process.env.BACKEND_PORT || 8080;
56
export const IP = process.env.IP || '0.0.0.0';
67
export const LOG_LEVEL = process.env.FASTIFY_LOG_LEVEL || process.env.LOG_LEVEL || 'info';
8+
export const LOG_DIR = path.join(__dirname, '../../../logs');
79
export const DEV_MODE = process.env.APP_ENV === 'development';
810
export const APP_ENV = process.env.APP_ENV;
911

backend/src/utils/fileUtils.ts

+61
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
import * as fs from 'fs';
2+
import { KubeFastifyInstance, OauthFastifyRequest } from '../types';
3+
import { LOG_DIR } from './constants';
4+
import { getUserName } from './userUtils';
5+
import { getNamespaces } from './notebookUtils';
6+
import { isUserAdmin } from './adminUtils';
7+
8+
export type AdminLogRecord = {
9+
user: string;
10+
namespace: string;
11+
action: string;
12+
endpoint: string;
13+
isAdmin: boolean;
14+
needsAdmin: boolean;
15+
};
16+
17+
export const logRequestDetails = (
18+
fastify: KubeFastifyInstance,
19+
request: OauthFastifyRequest,
20+
routeNeedsAdmin?: boolean,
21+
): void => {
22+
const data: Omit<AdminLogRecord, 'user' | 'isAdmin'> = {
23+
namespace: fastify.kube.namespace,
24+
action: request.method.toUpperCase(),
25+
endpoint: request.url.replace(request.headers.origin, ''),
26+
needsAdmin: routeNeedsAdmin ?? false,
27+
};
28+
29+
const writeLogAsync = async () => {
30+
const username = await getUserName(fastify, request);
31+
const { dashboardNamespace } = getNamespaces(fastify);
32+
const isAdmin = await isUserAdmin(fastify, username, dashboardNamespace);
33+
34+
writeAdminLog(fastify, {
35+
...data,
36+
user: username,
37+
isAdmin: isAdmin,
38+
});
39+
};
40+
// break the thread so the request is not held up logging / determing permissions of the user
41+
setTimeout(
42+
() => writeLogAsync().catch((e) => fastify.log.error(`Error writing log. ${e.message}`)),
43+
0,
44+
);
45+
};
46+
47+
export const writeAdminLog = (fastify: KubeFastifyInstance, data: AdminLogRecord): void => {
48+
try {
49+
fs.appendFile(
50+
`${LOG_DIR}/adminActivity.log`,
51+
`${new Date().toISOString()}: ${JSON.stringify(data)}\n`,
52+
function (err) {
53+
if (err) {
54+
fastify.log.error(`ERROR: Unable to write to admin log - ${err}`);
55+
}
56+
},
57+
);
58+
} catch (e) {
59+
fastify.log.error('Failed to log admin activity!');
60+
}
61+
};

backend/src/utils/route-security.ts

+3
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ import { createCustomError } from './requestUtils';
1010
import { FastifyReply, FastifyRequest } from 'fastify';
1111
import { isUserAdmin } from './adminUtils';
1212
import { getNamespaces } from './notebookUtils';
13+
import { logRequestDetails } from './fileUtils';
1314

1415
const testAdmin = async (
1516
fastify: KubeFastifyInstance,
@@ -153,6 +154,8 @@ const handleSecurityOnRouteData = async (
153154
request: OauthFastifyRequest,
154155
needsAdmin: boolean,
155156
): Promise<void> => {
157+
logRequestDetails(fastify, request, needsAdmin);
158+
156159
if (isRequestBody(request)) {
157160
await requestSecurityGuard(
158161
fastify,

0 commit comments

Comments
 (0)