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

Stop sending full defences on low levels #844

Merged
merged 43 commits into from
Mar 12, 2024
Merged
Show file tree
Hide file tree
Changes from 23 commits
Commits
Show all changes
43 commits
Select commit Hold shift + click to select a range
7bedff7
defences undefined for levels 1 and 2
pmarsh-scottlogic Feb 23, 2024
7e86ae4
fixes tests
pmarsh-scottlogic Feb 23, 2024
b0a4823
propoerly undefines defences on level 1 and 2
pmarsh-scottlogic Feb 23, 2024
4e45d90
moves variable declarations to destructureing
pmarsh-scottlogic Feb 29, 2024
d92314c
missed one
pmarsh-scottlogic Feb 29, 2024
74d8fb4
undo error swallow
pmarsh-scottlogic Feb 29, 2024
1237cca
merge dev
pmarsh-scottlogic Feb 29, 2024
dc66ffb
remove bad type guards
pmarsh-scottlogic Feb 29, 2024
c442d04
improve validation for configureDefence
pmarsh-scottlogic Feb 29, 2024
aa8f672
last bit of validation
pmarsh-scottlogic Feb 29, 2024
aa0c576
update tests for handleConfigureDefence
pmarsh-scottlogic Feb 29, 2024
d6b6455
new folder for defence controller unit tests and wrote first handleD…
pmarsh-scottlogic Feb 29, 2024
294de45
adds test for defence activation
pmarsh-scottlogic Feb 29, 2024
4785161
adds tests for defence deactivation
pmarsh-scottlogic Feb 29, 2024
66cdd2e
moves tests all back into one file again
pmarsh-scottlogic Feb 29, 2024
e74a26f
adds tests for handleResetSingleDefence
pmarsh-scottlogic Feb 29, 2024
663b22e
improve validation on handleGetDefenceStatus
pmarsh-scottlogic Feb 29, 2024
8429bfd
removes extraneous stuff from test objects
pmarsh-scottlogic Feb 29, 2024
ebba951
adds tests to make sure defences aren't chngeable for level 1
pmarsh-scottlogic Feb 29, 2024
107e453
add loop to test levels 1 and 2 with same code
pmarsh-scottlogic Feb 29, 2024
373e0b7
adds remaining tests for defence controller integration
pmarsh-scottlogic Feb 29, 2024
8293a4a
handleResetSingleDefence takes level parameter now
pmarsh-scottlogic Feb 29, 2024
ae5e746
adds level to frontend call to reset defence config item. Types all t…
pmarsh-scottlogic Feb 29, 2024
a336617
some renamings and fill in gaps in tests
pmarsh-scottlogic Feb 29, 2024
23771d9
adds another missing test
pmarsh-scottlogic Feb 29, 2024
2aab5f7
tweaks
pmarsh-scottlogic Feb 29, 2024
a04ca87
merge dev
pmarsh-scottlogic Mar 5, 2024
24b7c04
fix test suite
pmarsh-scottlogic Mar 5, 2024
2fc5a6f
replace defence search with defences.some and replace cuurrentDefence…
pmarsh-scottlogic Mar 5, 2024
e907b04
change defenceId for generic falsy check
pmarsh-scottlogic Mar 5, 2024
f56383f
improve guards for level and config
pmarsh-scottlogic Mar 5, 2024
65b5644
improve guards for level and config
pmarsh-scottlogic Mar 5, 2024
b45cce9
correct test for configuring a config item
pmarsh-scottlogic Mar 5, 2024
d670f00
updates error message when not allowed to activate or modify defences
pmarsh-scottlogic Mar 5, 2024
095ad15
rescturecure defence controller integration tests
pmarsh-scottlogic Mar 5, 2024
0d02968
change other tests to use test.each rather than foreach
pmarsh-scottlogic Mar 5, 2024
9f9d628
uses isValidLevel
pmarsh-scottlogic Mar 5, 2024
dec9904
improve validateFilterConfig
pmarsh-scottlogic Mar 5, 2024
e98c251
just pass qa defence rather than all defences
pmarsh-scottlogic Mar 7, 2024
b0e17d0
fix tests
pmarsh-scottlogic Mar 7, 2024
360c30c
uses type intersection to define QaLlmDefence
pmarsh-scottlogic Mar 12, 2024
600dc65
Merge branch 'dev' into stop-sending-full-defences-on-low-leevels
pmarsh-scottlogic Mar 12, 2024
8ec48c7
only gedDefencesFrDTOs if it is defined
pmarsh-scottlogic Mar 12, 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
169 changes: 117 additions & 52 deletions backend/src/controller/defenceController.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ import {
resetDefenceConfig,
} from '@src/defence';
import { DefenceActivateRequest } from '@src/models/api/DefenceActivateRequest';
import { DefenceConfigResetRequest } from '@src/models/api/DefenceConfigResetRequest';
import { DefenceConfigItemResetRequest } from '@src/models/api/DefenceConfigResetRequest';
import { DefenceConfigureRequest } from '@src/models/api/DefenceConfigureRequest';
import { DefenceStatusRequest } from '@src/models/api/DefenceStatusRequest';
import { DefenceConfigItem } from '@src/models/defence';
Expand All @@ -25,8 +25,7 @@ function configValueExceedsCharacterLimit(config: DefenceConfigItem[]) {
}

function handleDefenceActivation(req: DefenceActivateRequest, res: Response) {
const defenceId = req.body.defenceId;
const level = req.body.level;
const { defenceId, level } = req.body;

if (!defenceId) {
sendErrorResponse(res, 400, 'Missing defenceId');
Expand All @@ -43,7 +42,9 @@ function handleDefenceActivation(req: DefenceActivateRequest, res: Response) {
return;
}
pmarsh-scottlogic marked this conversation as resolved.
Show resolved Hide resolved

if (req.session.levelState[level].defences === undefined) {
const currentDefences = req.session.levelState[level].defences;

if (currentDefences === undefined) {
pmarsh-scottlogic marked this conversation as resolved.
Show resolved Hide resolved
sendErrorResponse(
res,
400,
Expand All @@ -52,18 +53,22 @@ function handleDefenceActivation(req: DefenceActivateRequest, res: Response) {
return;
}

if (req.session.levelState[3].defences === undefined) return; // this dumb type guard is needed, because we are accessing the array via enum, and so typescript doesn't account for the above type guards. This will be solved with #842. The below array indices won't need to be hardcoded either.
const defence = currentDefences.find((defence) => defence.id === defenceId);

if (defence === undefined) {
sendErrorResponse(res, 400, `Defence with id ${defenceId} not found`);
return;
}
pmarsh-scottlogic marked this conversation as resolved.
Show resolved Hide resolved

req.session.levelState[3].defences = activateDefence(
req.session.levelState[level].defences = activateDefence(
defenceId,
req.session.levelState[3].defences
currentDefences
);
res.status(200).send();
}

function handleDefenceDeactivation(req: DefenceActivateRequest, res: Response) {
const defenceId = req.body.defenceId;
const level = req.body.level;
const { defenceId, level } = req.body;

if (!defenceId) {
sendErrorResponse(res, 400, 'Missing defenceId');
Expand All @@ -80,7 +85,9 @@ function handleDefenceDeactivation(req: DefenceActivateRequest, res: Response) {
return;
}

if (req.session.levelState[level].defences === undefined) {
const currentDefences = req.session.levelState[level].defences;

if (currentDefences === undefined) {
sendErrorResponse(
res,
400,
Expand All @@ -89,23 +96,40 @@ function handleDefenceDeactivation(req: DefenceActivateRequest, res: Response) {
return;
}

if (req.session.levelState[3].defences === undefined) return; // this dumb type guard is needed, because we are accessing the array via enum, and so typescript doesn't account for the above type guards. This will be solved with #842. The below array indices won't need to be hardcoded either.
const defence = currentDefences.find((defence) => defence.id === defenceId);

if (defence === undefined) {
sendErrorResponse(res, 400, `Defence with id ${defenceId} not found`);
return;
}

req.session.levelState[3].defences = deactivateDefence(
req.session.levelState[level].defences = deactivateDefence(
defenceId,
req.session.levelState[3].defences
currentDefences
);
res.status(200).send();
}

function handleConfigureDefence(req: DefenceConfigureRequest, res: Response) {
// id of the defence
const defenceId = req.body.defenceId;
const config = req.body.config;
const level = req.body.level;
const { defenceId, level, config } = req.body;

if (defenceId === undefined) {
pmarsh-scottlogic marked this conversation as resolved.
Show resolved Hide resolved
sendErrorResponse(res, 400, 'Missing defenceId');
return;
}

if (level === undefined) {
chriswilty marked this conversation as resolved.
Show resolved Hide resolved
sendErrorResponse(res, 400, 'Missing level');
return;
}

if (level < LEVEL_NAMES.LEVEL_1 || level > LEVEL_NAMES.SANDBOX) {
sendErrorResponse(res, 400, 'Invalid level');
return;
}

if (!defenceId || !config || level === undefined) {
sendErrorResponse(res, 400, 'Missing defenceId, config or level');
if (config === undefined) {
pmarsh-scottlogic marked this conversation as resolved.
Show resolved Hide resolved
sendErrorResponse(res, 400, 'Missing config');
return;
}

Expand All @@ -114,7 +138,9 @@ function handleConfigureDefence(req: DefenceConfigureRequest, res: Response) {
return;
}

if (req.session.levelState[level].defences === undefined) {
const currentDefences = req.session.levelState[level].defences;

if (currentDefences === undefined) {
pmarsh-scottlogic marked this conversation as resolved.
Show resolved Hide resolved
sendErrorResponse(
res,
400,
Expand All @@ -123,77 +149,116 @@ function handleConfigureDefence(req: DefenceConfigureRequest, res: Response) {
return;
}

if (req.session.levelState[3].defences === undefined) return; // this dumb type guard is needed, because we are accessing the array via enum, and so typescript doesn't account for the above type guards. This will be solved with #842. The below array indices won't need to be hardcoded either.
const defence = currentDefences.find((defence) => defence.id === defenceId);

if (defence === undefined) {
sendErrorResponse(res, 400, `Defence with id ${defenceId} not found`);
return;
}

req.session.levelState[3].defences = configureDefence(
req.session.levelState[level].defences = configureDefence(
defenceId,
req.session.levelState[3].defences,
currentDefences,
config
);
res.send();
}

function handleResetSingleDefence(
req: DefenceConfigResetRequest,
function handleResetDefenceConfigItem(
req: DefenceConfigItemResetRequest,
res: Response
) {
const defenceId = req.body.defenceId;
const configId = req.body.configId;
const level = LEVEL_NAMES.SANDBOX; //configuration only available in sandbox (interesting that we force that here, but not in the above endpoints)
const { defenceId, configItemId, level } = req.body;

if (!defenceId) {
sendErrorResponse(res, 400, 'Missing defenceId');
return;
}

if (!configId) {
if (!configItemId) {
sendErrorResponse(res, 400, 'Missing configId');
return;
}

if (req.session.levelState[level].defences === undefined) {
if (level === undefined) {
sendErrorResponse(res, 400, 'Missing level');
return;
}

if (level < LEVEL_NAMES.LEVEL_1 || level > LEVEL_NAMES.SANDBOX) {
sendErrorResponse(res, 400, 'Invalid level');
return;
}

const currentDefences = req.session.levelState[level].defences;

if (currentDefences === undefined) {
sendErrorResponse(
res,
400,
'You cannot reset defence config items on this level, because it uses the default defences'
);
return;
}

const defence = currentDefences.find((defence) => defence.id === defenceId);

if (defence === undefined) {
sendErrorResponse(res, 400, `Defence with id ${defenceId} not found`);
return;
}

const configItem = defence.config.find(
(configItem) => configItem.id === configItemId
);

if (configItem === undefined) {
sendErrorResponse(
res,
400,
'You cannot reset defences on this level, because it uses the default defences'
`Config item with id ${configItemId} not found for defence with id ${defenceId}`
);
return;
}

req.session.levelState[level].defences = resetDefenceConfig(
defenceId,
configId,
req.session.levelState[level].defences
configItemId,
currentDefences
);
const updatedDefenceConfig: DefenceConfigItem | undefined =
req.session.levelState[level].defences
.find((defence) => defence.id === defenceId)
?.config.find((config) => config.id === configId);

if (updatedDefenceConfig) {
res.send(updatedDefenceConfig);
} else {
res.status(400);
res.send(
"something went wrong while resetting the defence's config. Check the defenceId and configId."
);

const updatedDefences = req.session.levelState[level].defences;
if (updatedDefences === undefined) {
sendErrorResponse(res, 500, 'Something went whacky');
return;
}
const updatedDefenceConfig = updatedDefences
.find((defence) => defence.id === defenceId)
?.config.find((config) => config.id === configItemId);

res.send(updatedDefenceConfig);
}

function handleGetDefenceStatus(req: DefenceStatusRequest, res: Response) {
const level: number | undefined = req.query.level as number | undefined;
if (level !== undefined) {
res.send(req.session.levelState[level].defences);
} else {
res.status(400);
res.send('Missing level');
const level = req.query.level;

if (level === undefined) {
sendErrorResponse(res, 400, 'Missing level');
return;
}

if (level < LEVEL_NAMES.LEVEL_1 || level > LEVEL_NAMES.SANDBOX) {
sendErrorResponse(res, 400, 'Invalid level');
return;
}

res.send(req.session.levelState[level].defences);
}

export {
handleDefenceActivation,
handleDefenceDeactivation,
handleConfigureDefence,
handleResetSingleDefence,
handleResetDefenceConfigItem,
handleGetDefenceStatus,
};
6 changes: 4 additions & 2 deletions backend/src/models/api/DefenceConfigResetRequest.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,13 +5,15 @@ import {
DEFENCE_ID,
DefenceConfigItem,
} from '@src/models/defence';
import { LEVEL_NAMES } from '@src/models/level';

export type DefenceConfigResetRequest = Request<
export type DefenceConfigItemResetRequest = Request<
never,
DefenceConfigItem,
{
defenceId?: DEFENCE_ID;
configId?: DEFENCE_CONFIG_ITEM_ID;
configItemId?: DEFENCE_CONFIG_ITEM_ID;
level?: LEVEL_NAMES;
},
never
>;
3 changes: 2 additions & 1 deletion backend/src/models/api/DefenceStatusRequest.ts
Original file line number Diff line number Diff line change
@@ -1,12 +1,13 @@
import { Request } from 'express';

import { Defence } from '@src/models/defence';
import { LEVEL_NAMES } from '@src/models/level';

export type DefenceStatusRequest = Request<
never,
Defence[] | string,
never,
{
level?: string;
level?: LEVEL_NAMES;
}
>;
25 changes: 14 additions & 11 deletions backend/src/models/level.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,19 +21,22 @@ interface LevelState {
}

function getInitialLevelStates() {
return Object.values(LEVEL_NAMES)
.filter((value) => Number.isNaN(Number(value)))
.map((value) => {
return {
level: value as LEVEL_NAMES,
const levelsWithDefences: number[] = [
LEVEL_NAMES.LEVEL_3,
LEVEL_NAMES.SANDBOX,
];

return Object.values(LEVEL_NAMES).map(
(level) =>
({
level,
chatHistory: [],
defences:
value === 'LEVEL_1' || value === 'LEVEL_2'
? undefined
: defaultDefences,
defences: levelsWithDefences.includes(level)
? defaultDefences
: undefined,
sentEmails: [],
} as LevelState;
});
} as LevelState)
);
}

export { getInitialLevelStates, LEVEL_NAMES };
Expand Down
3 changes: 1 addition & 2 deletions backend/src/server.ts
Original file line number Diff line number Diff line change
Expand Up @@ -24,8 +24,7 @@ app.listen(port, () => {
console.debug('Document vector store initialized');
})
.catch((err) => {
//throw new Error(`Error initializing document vectors: ${err}`);
console.log('ERROR SWALLOWED :D', err);
throw new Error(`Error initializing document vectors: ${err}`);
});

Promise.all([modelsPromise, vectorsPromise])
Expand Down
4 changes: 2 additions & 2 deletions backend/src/sessionRoutes.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ import {
handleDefenceActivation,
handleDefenceDeactivation,
handleGetDefenceStatus,
handleResetSingleDefence,
handleResetDefenceConfigItem,
} from './controller/defenceController';
import {
handleClearEmails,
Expand Down Expand Up @@ -102,7 +102,7 @@ router.get('/defence/status', handleGetDefenceStatus);
router.post('/defence/activate', handleDefenceActivation);
router.post('/defence/deactivate', handleDefenceDeactivation);
router.post('/defence/configure', handleConfigureDefence);
router.post('/defence/resetConfig', handleResetSingleDefence);
router.post('/defence/resetConfig', handleResetDefenceConfigItem);

// emails
router.get('/email/get', handleGetEmails);
Expand Down
Loading
Loading