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 39 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
16 changes: 12 additions & 4 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,10 +26,15 @@ on:
- enterprise
- personal
- none
tag:
description: 'tag the performance test'
required: true
default: 'Performance test'
type: string
vus:
description: 'Number of VUS'
required: true
default: 10
default: 1
type: number
duration:
description: 'Duration of test, ie 30s, 1m, 10m'
Expand All @@ -43,8 +48,11 @@ on:
type: choice
options:
- 'tests/k6/tests/serviceowner/performance/create-dialog.js'
- 'tests/k6/tests/enduser/performance/simple-search.js'
- 'tests/k6/tests/serviceowner/performance/create-remove-dialog.js'
- 'tests/k6/tests/enduser/performance/enduser-search.js'
- 'tests/k6/tests/graphql/performance/graphql-search.js'

run-name: ${{ inputs.tag }} vus ${{ inputs.vus }} duration ${{ inputs.duration }}
jobs:
k6-performance:
name: "Run K6 performance test"
Expand Down
1 change: 1 addition & 0 deletions .github/workflows/workflow-run-k6-performance.yml
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ on:
jobs:
k6-test:
runs-on: ubuntu-latest
environment: ${{ inputs.environment }}
permissions:
checks: write
pull-requests: write
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
27 changes: 27 additions & 0 deletions tests/k6/tests/enduser/performance/enduser-search.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
import { enduserSearch } from '../../performancetest_common/simpleSearch.js'
import { getDefaultThresholds } from '../../performancetest_common/getDefaultThresholds.js';
import { endUsersWithTokens } from '../../performancetest_common/readTestdata.js';
dagfinno marked this conversation as resolved.
Show resolved Hide resolved

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'],['enduser search',
'get dialog',
'get dialog activities',
'get dialog activity',
'get seenlogs',
'get seenlog',
'get transmissions',
'get transmission',
'get labellog'
])
};

export default function() {
if ((options.vus === undefined || options.vus === 1) && (options.iterations === undefined || options.iterations === 1)) {
enduserSearch(endUsersWithTokens[0]);
}
else {
enduserSearch(randomItem(endUsersWithTokens));
}
}
dagfinno marked this conversation as resolved.
Show resolved Hide resolved

57 changes: 0 additions & 57 deletions tests/k6/tests/enduser/performance/simple-search.js

This file was deleted.

23 changes: 23 additions & 0 deletions tests/k6/tests/graphql/performance/graphql-search.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
import { randomItem } from 'https://jslib.k6.io/k6-utils/1.2.0/index.js';
import { getDefaultThresholds } from '../../performancetest_common/getDefaultThresholds.js';
import { endUsersWithTokens as endUsers } from '../../performancetest_common/readTestdata.js';
import { graphqlSearch } from "../../performancetest_common/simpleSearch.js";

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));
}
}


53 changes: 53 additions & 0 deletions tests/k6/tests/performancetest_common/createDialog.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
import { uuidv4 } from 'https://jslib.k6.io/k6-utils/1.4.0/index.js';
import { describe } from "../../common/describe.js";
import { postSO, purgeSO } from "../../common/request.js";
import { expect } from "../../common/testimports.js";
import dialogToInsert from "../performancetest_data/01-create-dialog.js";

/**
* Creates a dialog.
*
* @param {Object} serviceOwner - The service owner.
* @param {Object} endUser - The end user.
*/

export function createDialog(serviceOwner, endUser) {
var paramsWithToken = {
headers: {
Authorization: "Bearer " + serviceOwner.token,
traceparent: uuidv4()
},
tags: { name: 'create dialog' }
};

describe('create dialog', () => {
let r = postSO('dialogs', dialogToInsert(endUser.ssn, endUser.resource), paramsWithToken);
expect(r.status, 'response status').to.equal(201);
});

}
dagfinno marked this conversation as resolved.
Show resolved Hide resolved

export function createAndRemoveDialog(serviceOwner, endUser) {
var paramsWithToken = {
headers: {
Authorization: "Bearer " + serviceOwner.token
},
tags: { name: 'create dialog' }
}

let dialogId = 0;
describe('create dialog', () => {
paramsWithToken.tags.name = 'create dialog';
let r = postSO('dialogs', dialogToInsert(endUser.ssn, endUser.resource), paramsWithToken);
expect(r.status, 'response status').to.equal(201);
dialogId = r.json();
});

describe('remove dialog', () => {
paramsWithToken.tags.name = 'remove dialog';
if (dialogId) {
let r = purgeSO('dialogs/' + dialogId, paramsWithToken);
expect(r.status, 'response status').to.equal(204);
}
});
}
dagfinno marked this conversation as resolved.
Show resolved Hide resolved
23 changes: 23 additions & 0 deletions tests/k6/tests/performancetest_common/getDefaultThresholds.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
/**
* 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;
}
21 changes: 21 additions & 0 deletions tests/k6/tests/performancetest_common/readTestdata.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
import papaparse from 'https://jslib.k6.io/papaparse/5.1.1/index.js';
import { SharedArray } from "k6/data";

const filenameServiceowners = '../performancetest_data/.serviceowners-with-tokens.csv';
dagfinno marked this conversation as resolved.
Show resolved Hide resolved
export const serviceOwners = new SharedArray('serviceOwners', function () {
return papaparse.parse(open(filenameServiceowners), { header: true, skipEmptyLines: true }).data;
});

if (!__ENV.API_ENVIRONMENT) {
throw new Error('API_ENVIRONMENT must be set');
}
const filenameEndusers = `../performancetest_data/endusers-${__ENV.API_ENVIRONMENT}.csv`;
export const endUsers = new SharedArray('endUsers', function () {
return papaparse.parse(open(filenameEndusers), { header: true, skipEmptyLines: true }).data;
});
dagfinno marked this conversation as resolved.
Show resolved Hide resolved

const filenameEndusersWithTokens = '../performancetest_data/.endusers-with-tokens.csv';
export const endUsersWithTokens = new SharedArray('endUsersWithTokens', function () {
return papaparse.parse(open(filenameEndusersWithTokens), { header: true, skipEmptyLines: true }).data;
});

Loading
Loading