Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

test: add more performance tests #1373

Merged
merged 43 commits into from
Nov 6, 2024
Merged
Show file tree
Hide file tree
Changes from 34 commits
Commits
Show all changes
43 commits
Select commit Hold shift + click to select a range
37a83d6
Added more tests
dagfinno Oct 31, 2024
0c2acae
add graphql endpoints
dagfinno Oct 31, 2024
d124a54
add POST request for graphql
dagfinno Oct 31, 2024
3f060c8
add POST request for graphql
dagfinno Oct 31, 2024
da0e851
add get for all enduser dialog paths
dagfinno Oct 31, 2024
2010d94
change performance to yt01
dagfinno Oct 31, 2024
fa65063
Use different create dialog mal
dagfinno Oct 31, 2024
3a94b76
simple search trough graphQL
dagfinno Oct 31, 2024
3f4753b
graphql post sample
dagfinno Oct 31, 2024
7a7b63d
endusers for yt01
dagfinno Oct 31, 2024
8200dc0
graphql post sample
dagfinno Oct 31, 2024
e213d25
serviceowner for yt01
dagfinno Oct 31, 2024
968eef3
test that runs create dialog and search in parallell
dagfinno Oct 31, 2024
401127d
test that first create and then delete a dialog
dagfinno Oct 31, 2024
cdcff0e
fixed filepath for graphql
dagfinno Oct 31, 2024
c677bd4
using dynamic name of workflow run
dagfinno Nov 1, 2024
2b9264d
using dynamic name of workflow run
dagfinno Nov 1, 2024
155fb44
using dynamic name of workflow run
dagfinno Nov 1, 2024
828851d
add http_req_failed rate to thresholds
dagfinno Nov 1, 2024
f4e5f0e
Merge branch 'main' into performance/create-tests
dagfinno Nov 1, 2024
fb08882
Merge branch 'main' into performance/create-tests
dagfinno Nov 1, 2024
cb037e4
chore: perf test url config refactor suggestion (#1375)
oskogstad Nov 1, 2024
80abedf
refactored to lower duplicated code rate
dagfinno Nov 1, 2024
2037132
Merge branch 'main' into performance/create-tests
dagfinno Nov 1, 2024
3af3e1e
Merge branch 'performance/create-tests' of github.com:digdir/dialogpo…
dagfinno Nov 1, 2024
28679dc
implements coderabbitai suggestions
dagfinno Nov 4, 2024
7e02e63
gets duplicated code rate down
dagfinno Nov 4, 2024
e7cf0da
gets duplicated code rate down
dagfinno Nov 4, 2024
74d678c
gets duplicated code rate down
dagfinno Nov 4, 2024
88b4b2b
gets duplicated code rate down
dagfinno Nov 4, 2024
8ff74c2
Merge branch 'main' into performance/create-tests
dagfinno Nov 4, 2024
6546395
implements suggestions from coderabbitai
dagfinno Nov 4, 2024
cc01bc5
implements suggestions from coderabbitai
dagfinno Nov 4, 2024
423962d
Merge branch 'performance/create-tests' of github.com:digdir/dialogpo…
dagfinno Nov 4, 2024
bd5bc8d
Merge branch 'main' into performance/create-tests
arealmaas Nov 5, 2024
d92b9ff
ci: ensure we use secrets from github secrets from env
arealmaas Nov 5, 2024
cbadc0b
Add traceparent header for otel
dagfinno Nov 5, 2024
bd0be61
refactoring; mainly moving tests and methods to new directories and f…
dagfinno Nov 5, 2024
088fc3c
refactoring, directory and file missed the previous commit
dagfinno Nov 5, 2024
5d13c27
documentation performed by github copilot
dagfinno Nov 5, 2024
87faf62
use newer version of k6-utils
dagfinno Nov 5, 2024
d19010b
Merge branch 'main' into performance/create-tests
dagfinno Nov 5, 2024
a864d86
Merge branch 'main' into performance/create-tests
dagfinno Nov 5, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 10 additions & 2 deletions .github/workflows/dispatch-k6-performance.yml
Original file line number Diff line number Diff line change
Expand Up @@ -10,12 +10,12 @@ on:
environment:
description: 'Environment'
required: true
default: 'staging'
default: 'yt01'
dagfinno marked this conversation as resolved.
Show resolved Hide resolved
type: choice
options:
- test
- staging
- performance
- yt01
tokens:
description: 'Tokens to generate; for create dialog, search, none, or both'
required: true
Expand All @@ -26,6 +26,11 @@ on:
- enterprise
- personal
- none
tag:
description: 'tag the performance test'
required: true
default: 'Performance test'
type: string
vus:
description: 'Number of VUS'
required: true
Expand All @@ -43,8 +48,11 @@ on:
type: choice
options:
- 'tests/k6/tests/serviceowner/performance/create-dialog.js'
- 'tests/k6/tests/serviceowner/performance/create-remove-dialog.js'
- 'tests/k6/tests/enduser/performance/simple-search.js'
- 'tests/k6/tests/enduser/performance/graphql-search.js'

run-name: ${{ inputs.tag }} vus ${{ inputs.vus }} duration ${{ inputs.duration }}
jobs:
k6-performance:
name: "Run K6 performance test"
Expand Down
49 changes: 35 additions & 14 deletions tests/k6/common/config.js
Original file line number Diff line number Diff line change
@@ -1,22 +1,41 @@
const localBaseUrl = "https://localhost:7214/";
const localDockerBaseUrl = "https://host.docker.internal:7214/";
const testBaseUrl = "https://altinn-dev-api.azure-api.net/dialogporten/";
const yt01BaseUrl = "https://platform.yt01.altinn.cloud/dialogporten/";
const stagingBaseUrl = "https://platform.tt02.altinn.no/dialogporten/";
const prodBaseUrl = "https://platform.altinn.no/dialogporten/";

const endUserPath = "api/v1/enduser/";
const serviceOwnerPath = "api/v1/serviceowner/";
const graphqlPath = "graphql";

export const baseUrls = {
v1: {
enduser: {
localdev: "https://localhost:7214/api/v1/enduser/",
localdev_docker: "https://host.docker.internal:7214/api/v1/enduser/",
test: "https://altinn-dev-api.azure-api.net/dialogporten/api/v1/enduser/",
yt01: "https://platform.yt01.altinn.cloud/dialogporten/api/v1/enduser/",
staging: "https://platform.tt02.altinn.no/dialogporten/api/v1/enduser/",
prod: "https://platform.altinn.no/dialogporten/api/v1/enduser/"
localdev: localBaseUrl + endUserPath,
localdev_docker: localDockerBaseUrl + endUserPath,
test: testBaseUrl + endUserPath,
yt01: yt01BaseUrl + endUserPath,
staging: stagingBaseUrl + endUserPath,
prod: prodBaseUrl + endUserPath
},
serviceowner: {
localdev: "https://localhost:7214/api/v1/serviceowner/",
localdev_docker: "https://host.docker.internal:7214/api/v1/serviceowner/",
test: "https://altinn-dev-api.azure-api.net/dialogporten/api/v1/serviceowner/",
yt01: "https://platform.yt01.altinn.cloud/dialogporten/api/v1/serviceowner/",
staging: "https://platform.tt02.altinn.no/dialogporten/api/v1/serviceowner/",
prod: "https://platform.altinn.no/dialogporten/api/v1/serviceowner/"
}
}
localdev: localBaseUrl + serviceOwnerPath,
localdev_docker: localDockerBaseUrl + serviceOwnerPath,
test: testBaseUrl + serviceOwnerPath,
yt01: yt01BaseUrl + serviceOwnerPath,
staging: stagingBaseUrl + serviceOwnerPath,
prod: prodBaseUrl + serviceOwnerPath
},
graphql: {
localdev: localBaseUrl + graphqlPath,
localdev_docker: localDockerBaseUrl + graphqlPath,
test: testBaseUrl + graphqlPath,
yt01: yt01BaseUrl + graphqlPath,
staging: stagingBaseUrl + graphqlPath,
prod: prodBaseUrl + graphqlPath
},
oskogstad marked this conversation as resolved.
Show resolved Hide resolved
}
};

export const defaultEndUserOrgNo = "310923044"; // ÆRLIG UROKKELIG TIGER AS
Expand All @@ -42,4 +61,6 @@ if (!baseUrls[__ENV.API_VERSION]["serviceowner"][__ENV.API_ENVIRONMENT]) {
export const baseUrlEndUser = baseUrls[__ENV.API_VERSION]["enduser"][__ENV.API_ENVIRONMENT];
export const baseUrlServiceOwner = baseUrls[__ENV.API_VERSION]["serviceowner"][__ENV.API_ENVIRONMENT];

export const baseUrlGraphql = baseUrls[__ENV.API_VERSION]["graphql"][__ENV.API_ENVIRONMENT];

dagfinno marked this conversation as resolved.
Show resolved Hide resolved
export const sentinelValue = "dialogporten-e2e-sentinel";
8 changes: 7 additions & 1 deletion tests/k6/common/request.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { default as http } from 'k6/http';
import { baseUrlEndUser, baseUrlServiceOwner } from './config.js'
import { baseUrlEndUser, baseUrlGraphql, baseUrlServiceOwner } from './config.js'
import { getServiceOwnerTokenFromGenerator, getEnduserTokenFromGenerator } from './token.js'
import { extend } from './extend.js'

Expand Down Expand Up @@ -125,3 +125,9 @@ export function patchEU(url, body, params = null, tokenOptions = null) {
export function deleteEU(url, params = null, tokenOptions = null) {
return http.request('DELETE', baseUrlEndUser + url, getEnduserRequestParams(params, tokenOptions));
}

export function postGQ(body, params = null) {
body = JSON.stringify({ query: body })
params = extend(true, {}, params, { headers: { 'Content-Type': 'application/json' }});
return http.post(baseUrlGraphql, body, params);
}
dagfinno marked this conversation as resolved.
Show resolved Hide resolved
3 changes: 2 additions & 1 deletion tests/k6/common/testimports.js
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,8 @@ export {
putSO,
patchSO,
deleteSO,
purgeSO
purgeSO,
postGQ
} from './request.js';
export {
setTitle,
Expand Down
42 changes: 42 additions & 0 deletions tests/k6/tests/enduser/performance/graphql-search.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
import { postGQ, expect, expectStatusFor, describe } from "../../../common/testimports.js";
import { randomItem } from 'https://jslib.k6.io/k6-utils/1.2.0/index.js';
import { getGraphqlParty } from '../../performancetest_data/graphql-search.js';
import { getEndusers, getDefaultThresholds } from '../../performancetest_common/common.js'


const filenameEndusers = __ENV.ENDUSERS_CSV_PATH || '../../performancetest_data/.endusers-with-tokens.csv';

const endUsers = getEndusers(filenameEndusers);

export let options = {
summaryTrendStats: ['avg', 'min', 'med', 'max', 'p(95)', 'p(99)', 'p(99.5)', 'p(99.9)', 'count'],
thresholds: getDefaultThresholds(['http_req_duration', 'http_reqs'],['graphql search'])
};

export default function() {
if (!endUsers || endUsers.length === 0) {
throw new Error('No end users loaded for testing');
}
if ((options.vus === undefined || options.vus === 1) && (options.iterations === undefined || options.iterations === 1)) {
graphqlSearch(endUsers[0]);
}
else {
graphqlSearch(randomItem(endUsers));
}
}

export function graphqlSearch(enduser) {
let paramsWithToken = {
headers: {
Authorization: "Bearer " + enduser.token,
},
tags: { name: 'graphql search' }
}
dagfinno marked this conversation as resolved.
Show resolved Hide resolved
describe('Perform graphql dialog list', () => {
//let r = ('dialogs' + defaultFilter, paramsWithToken);
let r = postGQ(getGraphqlParty(enduser.ssn), paramsWithToken);
expectStatusFor(r).to.equal(200);
expect(r, 'response').to.have.validJsonBody();
});
dagfinno marked this conversation as resolved.
Show resolved Hide resolved
}
dagfinno marked this conversation as resolved.
Show resolved Hide resolved

80 changes: 58 additions & 22 deletions tests/k6/tests/enduser/performance/simple-search.js
Original file line number Diff line number Diff line change
@@ -1,33 +1,23 @@
import { getEU, expect, expectStatusFor, describe } from "../../../common/testimports.js";
import { SharedArray } from 'k6/data';
import papaparse from 'https://jslib.k6.io/papaparse/5.1.1/index.js';
import { randomItem } from 'https://jslib.k6.io/k6-utils/1.2.0/index.js';
import { getEndusers, getDefaultThresholds } from '../../performancetest_common/common.js'

const filenameEndusers = '../../performancetest_data/.endusers-with-tokens.csv';

const endUsers = new SharedArray('endUsers', function () {
try {
const csvData = papaparse.parse(open(filenameEndusers), { header: true, skipEmptyLines: true }).data;
if (!csvData.length) {
throw new Error('No data found in CSV file');
}
csvData.forEach((user, index) => {
if (!user.token || !user.ssn) {
throw new Error(`Missing required fields at row ${index + 1}`);
}
});
return csvData;
} catch (error) {
throw new Error(`Failed to load end users: ${error.message}`);
}
});
const endUsers = getEndusers(filenameEndusers);

export let options = {
summaryTrendStats: ['avg', 'min', 'med', 'max', 'p(95)', 'p(99)', 'p(99.5)', 'p(99.9)', 'count'],
thresholds: {
'http_req_duration{name:simple search}': [],
'http_reqs{name:simple search}': [],
},
thresholds: getDefaultThresholds(['http_req_duration', 'http_reqs'],['simple search',
'get dialog',
'get dialog activities',
'get dialog activity',
'get seenlogs',
'get seenlog',
'get transmissions',
'get transmission',
'get labellog'
])
dagfinno marked this conversation as resolved.
Show resolved Hide resolved
};

export default function() {
Expand All @@ -39,6 +29,20 @@ export default function() {
}
}

function retrieveDialogContent(response, paramsWithToken) {
const items = response.json().items;
if (!items?.length) return;

const dialogId = items[0].id;
if (!dialogId) return;

getContent(dialogId, paramsWithToken, 'get dialog');
getContentChain(dialogId, paramsWithToken, 'get dialog activities', 'get dialog activity', '/activities/')
getContentChain(dialogId, paramsWithToken, 'get seenlogs', 'get seenlog', '/seenlog/')
getContent(dialogId, paramsWithToken, 'get labellog', '/labellog');
getContentChain(dialogId, paramsWithToken, 'get transmissions', 'get transmission', '/transmissions/')
}

export function simpleSearch(enduser) {
let paramsWithToken = {
headers: {
Expand All @@ -52,6 +56,38 @@ export function simpleSearch(enduser) {
let r = getEU('dialogs' + defaultFilter, paramsWithToken);
expectStatusFor(r).to.equal(200);
expect(r, 'response').to.have.validJsonBody();
retrieveDialogContent(r, paramsWithToken);
});
}

export function getContent(dialogId, paramsWithToken, tag, path = '') {
const listParams = {
...paramsWithToken,
tags: { ...paramsWithToken.tags, name: tag }
};
getUrl('dialogs/' + dialogId + path, listParams);
}

export function getContentChain(dialogId, paramsWithToken, tag, subtag, endpoint) {
const listParams = {
...paramsWithToken,
tags: { ...paramsWithToken.tags, name: tag }
};
let d = getUrl('dialogs/' + dialogId + endpoint, listParams);
let json = d.json();
if (json.length > 0) {
const detailParams = {
...paramsWithToken,
tags: { ...paramsWithToken.tags, name: subtag }
};
getUrl('dialogs/' + dialogId + endpoint + randomItem(json).id, detailParams);
}
}

export function getUrl(url, paramsWithToken) {
let r = getEU(url, paramsWithToken);
expectStatusFor(r).to.equal(200);
expect(r, 'response').to.have.validJsonBody();
return r;
}

51 changes: 51 additions & 0 deletions tests/k6/tests/performancetest_common/common.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
import { SharedArray } from 'k6/data';
import papaparse from 'https://jslib.k6.io/papaparse/5.1.1/index.js';

export function getEndusers(filenameEndusers) {
if (!filenameEndusers || typeof filenameEndusers !== 'string') {
throw new Error('filenameEndusers must be a non-empty string');
}

const endUsers = new SharedArray('endUsers', function () {
try {
const csvData = papaparse.parse(open(filenameEndusers), { header: true, skipEmptyLines: true }).data;
if (!csvData.length) {
throw new Error(`No data found in CSV file: ${filenameEndusers}`);
}
csvData.forEach((user, index) => {
if (!user.token || !user.ssn) {
throw new Error(`Missing required fields (token or ssn) at row ${index + 1} in ${filenameEndusers}`);
}
});
return csvData;
} catch (error) {
if (error.code === 'ENOENT') {
throw new Error(`CSV file not found: ${filenameEndusers}`);
}
throw new Error(`Failed to load end users from ${filenameEndusers}: ${error.message}`);
}
});
return endUsers;
}

/**
* Creates default thresholds configuration for K6 tests.
* @param {string[]} counters - Array of counter names
* @param {string[]} labels - Array of label names
* @returns {Object} Threshold configuration object
* @throws {Error} If inputs are invalid
*/
export function getDefaultThresholds(counters, labels) {
if (!Array.isArray(counters) || !Array.isArray(labels)) {
throw new Error('Both counters and labels must be arrays');
}
let thresholds = {
http_req_failed: ['rate<0.01']
}
for (const counter of counters) {
for (const label of labels) {
thresholds[`${counter}{name:${label}}`] = [];
}
}
return thresholds;
}
54 changes: 54 additions & 0 deletions tests/k6/tests/performancetest_data/01-create-dialog.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
import {default as createDialogPayload} from "../serviceowner/testdata/01-create-dialog.js"

const ACTIVITY_TYPE_INFORMATION = 'Information';

function cleanUp(originalPayload) {
if (!originalPayload || typeof originalPayload !== 'object') {
throw new Error('Invalid payload');
}

const payload = { ...originalPayload };
const { visibleFrom, ...payloadWithoutVisibleFrom } = payload;

const activities = payload.activities?.map(activity => {
if (activity.type !== ACTIVITY_TYPE_INFORMATION) {
return activity;
}

const { performedBy, ...rest } = activity;
const { actorId, ...performedByRest } = performedBy;

return {
...rest,
performedBy: {
...performedByRest,
actorName: "some name"
}
};
}) ?? [];

return {
...payloadWithoutVisibleFrom,
activities
};
}

/**
* Creates a dialog payload for performance testing
* @param {string} endUser - Norwegian national ID number (11 digits)
* @param {string} resource - Resource identifier
* @returns {Object} Dialog payload
* @throws {Error} If inputs are invalid
*/
export default function (endUser, resource) {
if (!endUser?.match(/^\d{11}$/)) {
throw new Error('endUser must be a 11-digit number');
}
if (!resource?.trim()) {
throw new Error('resource is required');
}
let payload = createDialogPayload();
payload.serviceResource = "urn:altinn:resource:" +resource;
payload.party = "urn:altinn:person:identifier-no:" + endUser;
return cleanUp(payload);
}
Loading
Loading