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

Feature/alignment poc #1174

Draft
wants to merge 127 commits into
base: main
Choose a base branch
from
Draft
Changes from 1 commit
Commits
Show all changes
127 commits
Select commit Hold shift + click to select a range
8043711
Finish backend implementation
lkleisa Feb 12, 2024
07e2db7
Finish frontend implementation
lkleisa Feb 12, 2024
cf19358
Delete alignment on kr or ob delete
lkleisa Feb 12, 2024
c5bef30
Write objectivecontroller tests
lkleisa Feb 12, 2024
1923850
Write ObjectiveAuthorizationService tests
lkleisa Feb 12, 2024
b1a2d95
Write AlignmentBusinessService tests
lkleisa Feb 12, 2024
1af132b
Write AlignmentPersistenceService tests
lkleisa Feb 12, 2024
e74760c
Fix AlignmentPersistenceService tests
lkleisa Feb 12, 2024
b91dbaa
Fix frontend model error
lkleisa Feb 12, 2024
8e87033
Write AlignmentValidationService tests
lkleisa Feb 15, 2024
428c28b
Write test for objective form component
lkleisa Feb 15, 2024
12763cc
Fix backend test
lkleisa Feb 15, 2024
a2ef67d
Fix backend tests
lkleisa Feb 15, 2024
f1db1b5
Write e2e tests
lkleisa Feb 15, 2024
1bf81f3
Fix e2e tests
lkleisa Feb 15, 2024
4e951e5
Adjust dropdown width
lkleisa Feb 15, 2024
988590c
Add null check for possibility length
lkleisa Feb 28, 2024
53640fa
Fix frontend test
lkleisa Feb 28, 2024
c865fbd
Fix e2e test
lkleisa Feb 28, 2024
f16244c
Remove alignment when objective quarter change
lkleisa Mar 14, 2024
026e62b
Add missing quarter in testdata and fix bug when editing alignment mu…
lkleisa Apr 25, 2024
6ca46c7
Add Kein Alignment for removing existing alignment
lkleisa Apr 25, 2024
f78d389
Fix backend tests
lkleisa Apr 25, 2024
abf2dff
Fix e2e tests because of new quarter
lkleisa Apr 25, 2024
082853a
Implement typeahead
lkleisa Apr 29, 2024
1a85d42
Sort alignment possibilities alphabetically
lkleisa May 23, 2024
ccc3327
Use bold font for optgroup label
lkleisa May 24, 2024
de88efc
Implement correct filter method for typeahead
lkleisa May 24, 2024
0f31d07
Fix bug with wrong quarter without queryParams
lkleisa May 24, 2024
64ca378
Add scrollLeft on input leave and adjust padding
lkleisa May 24, 2024
692b16e
Remove unused files
lkleisa May 24, 2024
8a0355e
Write backend tests
lkleisa May 27, 2024
427c57c
Clean up frontend
lkleisa May 27, 2024
33c6bf7
Adjust frontend tests
lkleisa May 27, 2024
a85272d
Adjust e2e tests
lkleisa May 27, 2024
436baa3
Fix frontend tests
lkleisa May 29, 2024
bb95ddd
Adjust filter function
lkleisa May 30, 2024
d6c192c
Fix bug with duplicated alignment possibilities
lkleisa Jun 3, 2024
c3be1ad
Simplify backend function
lkleisa Jun 3, 2024
b1cfa43
Add check for alignment in same team
lkleisa Jun 3, 2024
f54a884
Fix sonar code smells
lkleisa Jun 6, 2024
5c43a59
Clean up code
lkleisa Jun 6, 2024
719f14c
Remove AlignmentSelection
lkleisa Jun 7, 2024
cb2b1c4
Refactor code
lkleisa Jun 12, 2024
3d1deb3
[FM] Automated formating frontend
actions-user Jun 12, 2024
89b6967
Remove bug with duplicated alignment possibilities
lkleisa Jun 13, 2024
25de1a0
Use object for alignedEntity
lkleisa Jun 13, 2024
8b14f29
Rename alignmentObjects
lkleisa Jun 13, 2024
da14837
Add new migration with AlignmentView
lkleisa Apr 4, 2024
886befd
Add new model AlignmentView
lkleisa Apr 4, 2024
e40a56d
Add Repository and PersistenceService for AlignmentView
lkleisa Apr 4, 2024
0f22dbb
Add validateOnAlignmentGet Method in AlignmentValidationService
lkleisa Apr 4, 2024
77b618c
Use validation in AlignmentBusinessService
lkleisa Apr 4, 2024
1e75adc
Filter AlignmentViews which are not matching filter
lkleisa Apr 4, 2024
3657e6f
Add alignment data for local development
lkleisa Apr 4, 2024
77c6180
Add new DTOs to return compact data
lkleisa Apr 4, 2024
f571257
Add equals and hashCode for AlignmentView
lkleisa Apr 4, 2024
2e7cf72
Add alignment_view to db schema for testing
lkleisa Apr 4, 2024
12061d9
Implement function for getting Alignments by filters
lkleisa Apr 5, 2024
7ef5075
Add cytoscape.js and @types/cytoscape
lkleisa Apr 5, 2024
5fd014f
Implement tab-switch
lkleisa Apr 5, 2024
c6df3e4
Adjust uniqueId on AlignmentView for full uniqueness
lkleisa Apr 5, 2024
7bf2f11
Add AlignmentService in and get alignment data from filters
lkleisa Apr 5, 2024
74eea1b
Load alignment data on filter change
lkleisa Apr 5, 2024
b8eca02
Add new DiagramComponent and draw draft diagram with real data
lkleisa Apr 5, 2024
1b48731
Split up method in AlignmentBusinessService and adjust uniqueId for f…
lkleisa Apr 5, 2024
f10a5cc
Remove unused check in wrongAlignments filter
lkleisa Apr 5, 2024
50995e3
Add check for right counterpart
lkleisa Apr 8, 2024
d2edd03
Add SVG Styling for Diagram Nodes
lkleisa Apr 8, 2024
f0df281
Add user information when no alignments
lkleisa Apr 8, 2024
fc92b9c
Add diagram click handler
lkleisa Apr 8, 2024
2515eb0
Add missing quarter and move testdata to current quarter
lkleisa Apr 9, 2024
2ebf8db
Fix bug when no KeyResult in AlignmentData
lkleisa Apr 9, 2024
fd4c5ee
Write unit test for AlignmentBusinessService
lkleisa Apr 9, 2024
8b29edf
Write unit test for AlignmentValidationService
lkleisa Apr 9, 2024
8868e19
Write integration tests for AlignmentBusinessService
lkleisa Apr 9, 2024
6e68398
Adjust foreign key in testdata
lkleisa Apr 9, 2024
eac2b56
Write integration tests for AlignmentViewPersistenceService
lkleisa Apr 9, 2024
4eddd6d
Fix failing backend tests
lkleisa Apr 9, 2024
c714e5d
Add current quarter id to url on application launch
lkleisa Apr 9, 2024
50492f7
Fix frontend e2e tests
lkleisa Apr 9, 2024
967c969
Fix frontend unit tests for quarter-filter
lkleisa Apr 10, 2024
39a0e6c
Add frontend e2e tests for diagram
lkleisa Apr 10, 2024
73d7932
Rename function for replacing non ascii characters
lkleisa Apr 10, 2024
dec1ade
Write unit tests for DiagramComponent
lkleisa Apr 10, 2024
5045a7f
Add validation check for final AlignmentViewList on same amout of sou…
lkleisa Apr 10, 2024
fdfd82f
Write unit tests for new validation check
lkleisa Apr 10, 2024
1a07f52
Add translation for new backend error ALIGNMENT_DATA_FAIL
lkleisa Apr 10, 2024
3d56251
Add unit tests for AlignmentService
lkleisa Apr 11, 2024
e196b9f
Adjust height of commit icon in node
lkleisa Apr 11, 2024
fdfb206
Escape html tags in object title
lkleisa Apr 12, 2024
9d1a0e8
Return empty list when empty teamfilter is set
lkleisa Apr 22, 2024
8ce4ecb
Do not reload diagram when closing dialog with cancel
lkleisa Apr 22, 2024
edb59cc
Do not change data when click already active diagram tab
lkleisa Apr 22, 2024
506a8c5
Fix backend unit tests and e2e tests
lkleisa Apr 24, 2024
d10a05f
Remove reload of diagram when closing dialog
lkleisa Apr 24, 2024
562e1de
[FM] Automated formating frontend
actions-user Apr 29, 2024
d55a521
Use current quarter for alignments when no quarterFilter is set
lkleisa Apr 29, 2024
b7f7b62
Rename diagramm to network and k to kr
lkleisa May 6, 2024
eadc720
Add wheel sensitivity for diagram
lkleisa May 6, 2024
87636e5
Keep diagram on tab switch alive
lkleisa May 23, 2024
e4f4886
Remove icon on kr bubble
lkleisa May 23, 2024
eb30230
Adjust quarter fk in newquarterdata
lkleisa May 23, 2024
2e022b1
Adjust testdata after rebase
lkleisa May 30, 2024
9d3d3db
Adjust testdata after rebase
lkleisa May 31, 2024
17b6dc9
Change KR state on checkin edit
lkleisa May 31, 2024
3e223a3
Use another method to convert svg
lkleisa Jun 3, 2024
b557d9d
Fix failing tests
lkleisa Jun 3, 2024
36af8c0
[FM] Automated formating backend
actions-user Jun 3, 2024
9a729d4
Add hyphens to svg bubble on word break
lkleisa Jun 5, 2024
a2b0151
Clean up code
lkleisa Jun 5, 2024
111590c
Fix sonar code smells
lkleisa Jun 6, 2024
d87911f
Refactor code
lkleisa Jun 6, 2024
2ed1a3f
Adjust height of diagram
lkleisa Jun 6, 2024
5fd7ba6
[FM] Automated formating backend
actions-user Jun 13, 2024
9092638
Rename variable
lkleisa Jun 13, 2024
eab42af
Refactor frontend code
Jun 14, 2024
a8795f8
Use second refresh subject
lkleisa Jun 19, 2024
d15b26e
[FM] Automated formating backend
actions-user Nov 19, 2024
ed475a8
fix rebase errors
kcinay055679 Nov 19, 2024
08dde3e
fix backend tests by resolving import errors
kcinay055679 Nov 19, 2024
3271e2f
fix invalid quarters inside h2 test data
nevio18324 Nov 19, 2024
5a0973b
use proper quarterids for h2
kcinay055679 Nov 21, 2024
a40496a
add correct tenant in failing backend tests
kcinay055679 Nov 21, 2024
d745700
remove unused line in shared.module.ts
kcinay055679 Nov 21, 2024
0fcc86b
fix frontend tests
kcinay055679 Nov 21, 2024
4a8f6f2
add check for get last checkin ordinal
kcinay055679 Nov 21, 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
Prev Previous commit
Next Next commit
Write test for objective form component
lkleisa authored and kcinay055679 committed Nov 21, 2024
commit 428c28bcac1935141debb8abaab1c632dace53cb
Original file line number Diff line number Diff line change
@@ -73,7 +73,6 @@
</ng-container>
</select>
</div>
<h1>{{ this.objectiveForm.value.alignment }}</h1>
</div>
</div>
<div class="col align-self-end">
Original file line number Diff line number Diff line change
@@ -9,7 +9,7 @@ import { MatSelectModule } from '@angular/material/select';
import { ReactiveFormsModule } from '@angular/forms';
import { NoopAnimationsModule } from '@angular/platform-browser/animations';
import { ObjectiveService } from '../../../services/objective.service';
import { marketingTeamWriteable, objective, quarter, quarterList } from '../../testData';
import { marketingTeamWriteable, objective, objectiveWithAlignment, quarter, quarterList } from '../../testData';
import { Observable, of } from 'rxjs';
import { TestbedHarnessEnvironment } from '@angular/cdk/testing/testbed';
import { HarnessLoader } from '@angular/cdk/testing';
@@ -36,6 +36,7 @@ let objectiveService = {
createObjective: jest.fn(),
updateObjective: jest.fn(),
deleteObjective: jest.fn(),
getAlignmentPossibilities: jest.fn(),
};

interface MatDialogDataInterface {
@@ -83,6 +84,24 @@ const mockActivatedRoute = {
},
};

const alignmentPossibilities = [
{
objectiveId: 1003,
objectiveTitle: 'O - Test Objective',
keyResultAlignmentsDtos: [],
},
{
objectiveId: 1005,
objectiveTitle: 'O - Company will grow',
keyResultAlignmentsDtos: [
{
keyResultId: 6,
keyResultTitle: 'K - New structure',
},
],
},
];

describe('ObjectiveDialogComponent', () => {
let component: ObjectiveFormComponent;
let fixture: ComponentFixture<ObjectiveFormComponent>;
@@ -117,6 +136,7 @@ describe('ObjectiveDialogComponent', () => {
{ provide: TeamService, useValue: teamService },
],
});
jest.spyOn(objectiveService, 'getAlignmentPossibilities').mockReturnValue(of(alignmentPossibilities));
fixture = TestBed.createComponent(ObjectiveFormComponent);
component = fixture.componentInstance;
fixture.detectChanges();
@@ -172,23 +192,24 @@ describe('ObjectiveDialogComponent', () => {
objectiveService.createObjective.mockReturnValue(of({ ...objective, state: state }));
component.onSubmit(state);

expect(dialogMock.close).toHaveBeenCalledWith({
addKeyResult: createKeyresults,
delete: false,
objective: {
description: description,
id: 5,
version: 1,
quarterId: 2,
quarterLabel: 'GJ 22/23-Q2',
state: State[state as keyof typeof State],
teamId: 2,
title: title,
writeable: true,
},
teamId: 1,
});
}),
expect(dialogMock.close).toHaveBeenCalledWith({
addKeyResult: createKeyresults,
delete: false,
objective: {
description: description,
id: 5,
version: 1,
quarterId: 2,
quarterLabel: 'GJ 22/23-Q2',
state: State[state as keyof typeof State],
teamId: 2,
title: title,
writeable: true,
alignedEntityId: null,
},
teamId: 1,
});
}),
);

it('should create objective', () => {
@@ -198,7 +219,7 @@ describe('ObjectiveDialogComponent', () => {
description: 'Test description',
quarter: 0,
team: 0,
relation: 0,
alignment: '',
createKeyResults: false,
});

@@ -214,6 +235,36 @@ describe('ObjectiveDialogComponent', () => {
title: 'Test title',
quarterId: 0,
teamId: 0,
version: undefined,
alignedEntityId: '',
});
});

it('should create objective with alignment', () => {
matDataMock.objective.objectiveId = undefined;
component.objectiveForm.setValue({
title: 'Test title with alignment',
description: 'Test description',
quarter: 0,
team: 0,
alignment: 'K37',
createKeyResults: false,
});

objectiveService.createObjective.mockReturnValue(of({ ...objective, state: 'DRAFT' }));
component.onSubmit('DRAFT');

fixture.detectChanges();

expect(objectiveService.createObjective).toHaveBeenCalledWith({
description: 'Test description',
id: undefined,
state: 'DRAFT',
title: 'Test title with alignment',
quarterId: 0,
teamId: 0,
version: undefined,
alignedEntityId: 'K37',
});
});

@@ -224,7 +275,7 @@ describe('ObjectiveDialogComponent', () => {
description: 'Test description',
quarter: 1,
team: 1,
relation: 0,
alignment: '',
createKeyResults: false,
});

@@ -240,6 +291,36 @@ describe('ObjectiveDialogComponent', () => {
title: 'Test title',
quarterId: 1,
teamId: 1,
version: undefined,
alignedEntityId: '',
});
});

it('should update objective with alignment', () => {
matDataMock.objective.objectiveId = 1;
component.objectiveForm.setValue({
title: 'Test title with alignment',
description: 'Test description',
quarter: 1,
team: 1,
alignment: 'K37',
createKeyResults: false,
});

objectiveService.updateObjective.mockReturnValue(of({ ...objective, state: 'ONGOING' }));
component.onSubmit('DRAFT');

fixture.detectChanges();

expect(objectiveService.updateObjective).toHaveBeenCalledWith({
description: 'Test description',
id: 1,
state: 'DRAFT',
title: 'Test title with alignment',
quarterId: 1,
teamId: 1,
version: undefined,
alignedEntityId: 'K37',
});
});

@@ -270,6 +351,20 @@ describe('ObjectiveDialogComponent', () => {
expect(rawFormValue.quarter).toBe(objective.quarterId);
});

it('should load default values into form onInit with defined objectiveId with an alignment', async () => {
matDataMock.objective.objectiveId = 1;
const routerHarness = await RouterTestingHarness.create();
await routerHarness.navigateByUrl('/?quarter=2');
objectiveService.getFullObjective.mockReturnValue(of(objectiveWithAlignment));
component.ngOnInit();
const rawFormValue = component.objectiveForm.getRawValue();
expect(rawFormValue.title).toBe(objectiveWithAlignment.title);
expect(rawFormValue.description).toBe(objectiveWithAlignment.description);
expect(rawFormValue.team).toBe(objectiveWithAlignment.teamId);
expect(rawFormValue.quarter).toBe(objectiveWithAlignment.quarterId);
expect(rawFormValue.alignment).toBe(objectiveWithAlignment.alignedEntityId);
});

it('should return correct value if allowed to save to backlog', async () => {
component.quarters = quarterList;
const isBacklogQuarterSpy = jest.spyOn(component, 'isBacklogQuarter');
@@ -344,6 +439,80 @@ describe('ObjectiveDialogComponent', () => {
fixture.detectChanges();
expect(component.allowedOption(quarter)).toBeTruthy();
});

it('should load correct alignment possibilities', async () => {
let generatedPossibilities = [
{
objectiveId: null,
objectiveTitle: 'Bitte wählen',
keyResultAlignmentsDtos: [],
},
{
objectiveId: 1003,
objectiveTitle: 'O - Test Objective',
keyResultAlignmentsDtos: [],
},
{
objectiveId: 1005,
objectiveTitle: 'O - Company will grow',
keyResultAlignmentsDtos: [
{
keyResultId: 6,
keyResultTitle: 'K - New structure',
},
],
},
];

let componentValue = null;
component.alignmentPossibilities$.subscribe((value) => {
componentValue = value;
});
expect(componentValue).toStrictEqual(generatedPossibilities);
});

it('should not load current Objective to alignment possibilities', async () => {
matDataMock.objective.objectiveId = 1;
component.objective = objectiveWithAlignment;
const routerHarness = await RouterTestingHarness.create();
await routerHarness.navigateByUrl('/?quarter=2');
objectiveService.getFullObjective.mockReturnValue(of(objectiveWithAlignment));
component.ngOnInit();

let generatedPossibilities = [
{
objectiveId: null,
objectiveTitle: 'Bitte wählen',
keyResultAlignmentsDtos: [],
},
{
objectiveId: 1003,
objectiveTitle: 'O - Test Objective',
keyResultAlignmentsDtos: [],
},
{
objectiveId: 1005,
objectiveTitle: 'O - Company will grow',
keyResultAlignmentsDtos: [
{
keyResultId: 6,
keyResultTitle: 'K - New structure',
},
],
},
];

let componentValue = null;
component.alignmentPossibilities$.subscribe((value) => {
componentValue = value;
});
expect(componentValue).toStrictEqual(generatedPossibilities);
});

it('should call ObjectiveService when updating Alignments', async () => {
component.updateAlignments();
expect(objectiveService.getAlignmentPossibilities).toHaveBeenCalled();
});
});

describe('Backlog quarter', () => {
@@ -376,6 +545,7 @@ describe('ObjectiveDialogComponent', () => {
{ provide: ActivatedRoute, useValue: mockActivatedRoute },
],
});
jest.spyOn(objectiveService, 'getAlignmentPossibilities').mockReturnValue(of(alignmentPossibilities));
fixture = TestBed.createComponent(ObjectiveFormComponent);
component = fixture.componentInstance;
fixture.detectChanges();
Original file line number Diff line number Diff line change
@@ -185,6 +185,7 @@ export class ObjectiveFormComponent implements OnInit, OnDestroy {
state: 'DRAFT' as State,
teamId: 0,
quarterId: 0,
alignedEntityId: null,
} as Objective;
}

@@ -250,12 +251,14 @@ export class ObjectiveFormComponent implements OnInit, OnDestroy {
if (this.objective?.id) {
value = value.filter((item) => !(item.objectiveId == this.objective!.id));
}
let noSelectOption: AlignmentPossibility = {
objectiveId: null,
objectiveTitle: 'Bitte wählen',
keyResultAlignmentsDtos: [],
};
value.unshift(noSelectOption);
if (value[0].objectiveTitle != 'Bitte wählen') {
let noSelectOption: AlignmentPossibility = {
objectiveId: null,
objectiveTitle: 'Bitte wählen',
keyResultAlignmentsDtos: [],
};
value.unshift(noSelectOption);
}
this.alignmentPossibilities$ = of(value);
});
}
13 changes: 13 additions & 0 deletions frontend/src/app/shared/testData.ts
Original file line number Diff line number Diff line change
@@ -322,6 +322,19 @@ export const objective: Objective = {
alignedEntityId: null,
};

export const objectiveWithAlignment: Objective = {
id: 5,
version: 1,
title: 'title',
description: 'description',
teamId: 2,
quarterId: 2,
quarterLabel: 'GJ 22/23-Q2',
state: State.SUCCESSFUL,
writeable: true,
alignedEntityId: 'O6',
};

export const objectiveWriteableFalse: Objective = {
id: 6,
version: 1,
2 changes: 2 additions & 0 deletions frontend/src/assets/i18n/de.json
Original file line number Diff line number Diff line change
@@ -91,6 +91,8 @@
"NOT_AUTHORIZED_TO_DELETE": "Du bist nicht autorisiert, um dieses {0} zu löschen.",
"TOKEN_NULL": "Das erhaltene Token ist null.",
"ILLEGAL_CHANGE_OBJECTIVE_QUARTER": "Element kann nicht in ein anderes Quartal verlegt werden.",
"NOT_LINK_YOURSELF": "Das Objective kann nicht auf sich selbst zeigen.",
"ALIGNMENT_ALREADY_EXISTS": "Es existiert bereits ein Alignment ausgehend vom Objective mit der ID {1}."
"TRIED_TO_DELETE_LAST_ADMIN": "Der letzte Administrator eines Teams kann nicht entfernt werden",
"TRIED_TO_REMOVE_LAST_OKR_CHAMPION": "Der letzte OKR Champion kann nicht entfernt werden"
},