feat: NoValue is bot properly created the backend (#9110)

`No Value` view groups wasn't properly created when we select a group by
field metadata, this PR fix the issue.
Also a script is added to backfill the current view groups.

---------

Co-authored-by: Marie <51697796+ijreilly@users.noreply.github.com>
This commit is contained in:
Jérémy M 2024-12-18 12:26:38 +01:00 committed by GitHub
parent d895468ebe
commit 3b48920314
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
7 changed files with 178 additions and 53 deletions

View File

@ -73,6 +73,20 @@ export const useHandleRecordGroupField = ({
}) satisfies ViewGroup, }) satisfies ViewGroup,
); );
if (
!existingGroupKeys.has(`${fieldMetadataItem.id}:`) &&
fieldMetadataItem.isNullable === true
) {
viewGroupsToCreate.push({
__typename: 'ViewGroup',
id: v4(),
fieldValue: '',
isVisible: true,
position: fieldMetadataItem.options.length,
fieldMetadataId: fieldMetadataItem.id,
} satisfies ViewGroup);
}
const viewGroupsToDelete = view.viewGroups.filter( const viewGroupsToDelete = view.viewGroups.filter(
(group) => group.fieldMetadataId !== fieldMetadataItem.id, (group) => group.fieldMetadataId !== fieldMetadataItem.id,
); );

View File

@ -41,46 +41,25 @@ export const mapViewGroupsToRecordGroupDefinitions = ({
(option) => option.value === viewGroup.fieldValue, (option) => option.value === viewGroup.fieldValue,
); );
if (!selectedOption) { if (!selectedOption && selectFieldMetadataItem.isNullable === false) {
return null; return null;
} }
return { return {
id: viewGroup.id, id: viewGroup.id,
fieldMetadataId: viewGroup.fieldMetadataId, fieldMetadataId: viewGroup.fieldMetadataId,
type: RecordGroupDefinitionType.Value, type: !isDefined(selectedOption)
title: selectedOption.label, ? RecordGroupDefinitionType.NoValue
value: selectedOption.value, : RecordGroupDefinitionType.Value,
color: selectedOption.color, title: selectedOption?.label ?? 'No Value',
value: selectedOption?.value ?? null,
color: selectedOption?.color ?? 'transparent',
position: viewGroup.position, position: viewGroup.position,
isVisible: viewGroup.isVisible, isVisible: viewGroup.isVisible,
} as RecordGroupDefinition; } as RecordGroupDefinition;
}) })
.filter(isDefined); .filter(isDefined);
if (selectFieldMetadataItem.isNullable === true) {
const viewGroup = viewGroups.find(
(viewGroup) => viewGroup.fieldValue === '',
);
const noValueColumn = {
id: viewGroup?.id ?? '20202020-c05f-46c9-ae1e-2b3c5c702049',
title: 'No Value',
type: RecordGroupDefinitionType.NoValue,
value: null,
position:
viewGroup?.position ??
recordGroupDefinitionsFromViewGroups
.map((option) => option.position)
.reduce((a, b) => Math.max(a, b), 0) + 1,
isVisible: viewGroup?.isVisible ?? true,
fieldMetadataId: selectFieldMetadataItem.id,
color: 'transparent',
} satisfies RecordGroupDefinition;
return [...recordGroupDefinitionsFromViewGroups, noValueColumn];
}
return recordGroupDefinitionsFromViewGroups.sort( return recordGroupDefinitionsFromViewGroups.sort(
(a, b) => a.position - b.position, (a, b) => a.position - b.position,
); );

View File

@ -6,6 +6,7 @@ import { Repository } from 'typeorm';
import { ActiveWorkspacesCommandRunner } from 'src/database/commands/active-workspaces.command'; import { ActiveWorkspacesCommandRunner } from 'src/database/commands/active-workspaces.command';
import { BaseCommandOptions } from 'src/database/commands/base.command'; import { BaseCommandOptions } from 'src/database/commands/base.command';
import { RecordPositionBackfillCommand } from 'src/database/commands/upgrade-version/0-40/0-40-record-position-backfill.command'; import { RecordPositionBackfillCommand } from 'src/database/commands/upgrade-version/0-40/0-40-record-position-backfill.command';
import { ViewGroupNoValueBackfillCommand } from 'src/database/commands/upgrade-version/0-40/0-40-view-group-no-value-backfill.command';
import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity';
import { SyncWorkspaceMetadataCommand } from 'src/engine/workspace-manager/workspace-sync-metadata/commands/sync-workspace-metadata.command'; import { SyncWorkspaceMetadataCommand } from 'src/engine/workspace-manager/workspace-sync-metadata/commands/sync-workspace-metadata.command';
@ -18,6 +19,7 @@ export class UpgradeTo0_40Command extends ActiveWorkspacesCommandRunner {
@InjectRepository(Workspace, 'core') @InjectRepository(Workspace, 'core')
protected readonly workspaceRepository: Repository<Workspace>, protected readonly workspaceRepository: Repository<Workspace>,
private readonly recordPositionBackfillCommand: RecordPositionBackfillCommand, private readonly recordPositionBackfillCommand: RecordPositionBackfillCommand,
private readonly viewGroupNoValueBackfillCommand: ViewGroupNoValueBackfillCommand,
private readonly syncWorkspaceMetadataCommand: SyncWorkspaceMetadataCommand, private readonly syncWorkspaceMetadataCommand: SyncWorkspaceMetadataCommand,
) { ) {
super(workspaceRepository); super(workspaceRepository);
@ -34,6 +36,12 @@ export class UpgradeTo0_40Command extends ActiveWorkspacesCommandRunner {
workspaceIds, workspaceIds,
); );
await this.viewGroupNoValueBackfillCommand.executeActiveWorkspacesCommand(
passedParam,
options,
workspaceIds,
);
await this.syncWorkspaceMetadataCommand.executeActiveWorkspacesCommand( await this.syncWorkspaceMetadataCommand.executeActiveWorkspacesCommand(
passedParam, passedParam,
{ {

View File

@ -3,16 +3,25 @@ import { TypeOrmModule } from '@nestjs/typeorm';
import { RecordPositionBackfillCommand } from 'src/database/commands/upgrade-version/0-40/0-40-record-position-backfill.command'; import { RecordPositionBackfillCommand } from 'src/database/commands/upgrade-version/0-40/0-40-record-position-backfill.command';
import { UpgradeTo0_40Command } from 'src/database/commands/upgrade-version/0-40/0-40-upgrade-version.command'; import { UpgradeTo0_40Command } from 'src/database/commands/upgrade-version/0-40/0-40-upgrade-version.command';
import { ViewGroupNoValueBackfillCommand } from 'src/database/commands/upgrade-version/0-40/0-40-view-group-no-value-backfill.command';
import { RecordPositionBackfillModule } from 'src/engine/api/graphql/workspace-query-runner/services/record-position-backfill-module'; import { RecordPositionBackfillModule } from 'src/engine/api/graphql/workspace-query-runner/services/record-position-backfill-module';
import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity';
import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity';
import { FieldMetadataModule } from 'src/engine/metadata-modules/field-metadata/field-metadata.module';
import { WorkspaceSyncMetadataCommandsModule } from 'src/engine/workspace-manager/workspace-sync-metadata/commands/workspace-sync-metadata-commands.module'; import { WorkspaceSyncMetadataCommandsModule } from 'src/engine/workspace-manager/workspace-sync-metadata/commands/workspace-sync-metadata-commands.module';
@Module({ @Module({
imports: [ imports: [
TypeOrmModule.forFeature([Workspace], 'core'), TypeOrmModule.forFeature([Workspace], 'core'),
TypeOrmModule.forFeature([FieldMetadataEntity], 'metadata'),
WorkspaceSyncMetadataCommandsModule, WorkspaceSyncMetadataCommandsModule,
RecordPositionBackfillModule, RecordPositionBackfillModule,
FieldMetadataModule,
],
providers: [
UpgradeTo0_40Command,
RecordPositionBackfillCommand,
ViewGroupNoValueBackfillCommand,
], ],
providers: [UpgradeTo0_40Command, RecordPositionBackfillCommand],
}) })
export class UpgradeTo0_40CommandModule {} export class UpgradeTo0_40CommandModule {}

View File

@ -0,0 +1,82 @@
import { InjectRepository } from '@nestjs/typeorm';
import { Command } from 'nest-commander';
import { Repository } from 'typeorm';
import { ActiveWorkspacesCommandRunner } from 'src/database/commands/active-workspaces.command';
import { BaseCommandOptions } from 'src/database/commands/base.command';
import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity';
import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity';
import { FieldMetadataRelatedRecordsService } from 'src/engine/metadata-modules/field-metadata/services/field-metadata-related-records.service';
import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
import { ViewGroupWorkspaceEntity } from 'src/modules/view/standard-objects/view-group.workspace-entity';
import { ViewWorkspaceEntity } from 'src/modules/view/standard-objects/view.workspace-entity';
@Command({
name: 'migrate-0.40:backfill-view-group-no-value',
description: 'Backfill view group no value',
})
export class ViewGroupNoValueBackfillCommand extends ActiveWorkspacesCommandRunner {
constructor(
@InjectRepository(Workspace, 'core')
protected readonly workspaceRepository: Repository<Workspace>,
@InjectRepository(FieldMetadataEntity, 'metadata')
private readonly fieldMetadataRepository: Repository<FieldMetadataEntity>,
private readonly twentyORMGlobalManager: TwentyORMGlobalManager,
private readonly fieldMetadataRelatedRecordsService: FieldMetadataRelatedRecordsService,
) {
super(workspaceRepository);
}
async executeActiveWorkspacesCommand(
_passedParam: string[],
_options: BaseCommandOptions,
workspaceIds: string[],
): Promise<void> {
for (const workspaceId of workspaceIds) {
const viewRepository =
await this.twentyORMGlobalManager.getRepositoryForWorkspace<ViewWorkspaceEntity>(
workspaceId,
'view',
);
const viewGroupRepository =
await this.twentyORMGlobalManager.getRepositoryForWorkspace<ViewGroupWorkspaceEntity>(
workspaceId,
'viewGroup',
);
const views = await viewRepository.find({
relations: ['viewGroups'],
});
for (const view of views) {
if (view.viewGroups.length === 0) {
continue;
}
// We're assuming for now that all viewGroups belonging to the same view have the same fieldMetadataId
const viewGroup = view.viewGroups?.[0];
const fieldMetadataId = viewGroup?.fieldMetadataId;
if (!fieldMetadataId || !viewGroup) {
continue;
}
const fieldMetadata = await this.fieldMetadataRepository.findOne({
where: { id: viewGroup.fieldMetadataId },
});
if (!fieldMetadata) {
continue;
}
await this.fieldMetadataRelatedRecordsService.syncNoValueViewGroup(
fieldMetadata,
view,
viewGroupRepository,
);
}
}
}
}

View File

@ -84,8 +84,9 @@ import { UpdateFieldInput } from './dtos/update-field.input';
IsFieldMetadataDefaultValue, IsFieldMetadataDefaultValue,
IsFieldMetadataOptions, IsFieldMetadataOptions,
FieldMetadataService, FieldMetadataService,
FieldMetadataRelatedRecordsService,
FieldMetadataResolver, FieldMetadataResolver,
], ],
exports: [FieldMetadataService], exports: [FieldMetadataService, FieldMetadataRelatedRecordsService],
}) })
export class FieldMetadataModule {} export class FieldMetadataModule {}

View File

@ -8,6 +8,7 @@ import {
} from 'src/engine/metadata-modules/field-metadata/dtos/options.input'; } from 'src/engine/metadata-modules/field-metadata/dtos/options.input';
import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity';
import { isSelectFieldMetadataType } from 'src/engine/metadata-modules/field-metadata/utils/is-select-field-metadata-type.util'; import { isSelectFieldMetadataType } from 'src/engine/metadata-modules/field-metadata/utils/is-select-field-metadata-type.util';
import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository';
import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager'; import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
import { ViewGroupWorkspaceEntity } from 'src/modules/view/standard-objects/view-group.workspace-entity'; import { ViewGroupWorkspaceEntity } from 'src/modules/view/standard-objects/view-group.workspace-entity';
import { ViewWorkspaceEntity } from 'src/modules/view/standard-objects/view.workspace-entity'; import { ViewWorkspaceEntity } from 'src/modules/view/standard-objects/view.workspace-entity';
@ -28,7 +29,7 @@ export class FieldMetadataRelatedRecordsService {
oldFieldMetadata: FieldMetadataEntity, oldFieldMetadata: FieldMetadataEntity,
newFieldMetadata: FieldMetadataEntity, newFieldMetadata: FieldMetadataEntity,
transactionManager?: EntityManager, transactionManager?: EntityManager,
) { ): Promise<void> {
if ( if (
!isSelectFieldMetadataType(newFieldMetadata.type) || !isSelectFieldMetadataType(newFieldMetadata.type) ||
!isSelectFieldMetadataType(oldFieldMetadata.type) !isSelectFieldMetadataType(oldFieldMetadata.type)
@ -54,10 +55,7 @@ export class FieldMetadataRelatedRecordsService {
continue; continue;
} }
const maxPosition = view.viewGroups.reduce( const maxPosition = this.getMaxPosition(view.viewGroups);
(max, viewGroup) => Math.max(max, viewGroup.position),
0,
);
const viewGroupsToCreate = created.map((option, index) => const viewGroupsToCreate = created.map((option, index) =>
viewGroupRepository.create({ viewGroupRepository.create({
@ -72,21 +70,19 @@ export class FieldMetadataRelatedRecordsService {
await viewGroupRepository.insert(viewGroupsToCreate, transactionManager); await viewGroupRepository.insert(viewGroupsToCreate, transactionManager);
for (const { old: oldOption, new: newOption } of updated) { for (const { old: oldOption, new: newOption } of updated) {
const viewGroup = view.viewGroups.find( const existingViewGroup = view.viewGroups.find(
(viewGroup) => viewGroup.fieldValue === oldOption.value, (group) => group.fieldValue === oldOption.value,
); );
if (!viewGroup) { if (!existingViewGroup) {
throw new Error(`View group not found for option ${oldOption.value}`); throw new Error(
`View group not found for option "${oldOption.value}" during update.`,
);
} }
await viewGroupRepository.update( await viewGroupRepository.update(
{ { id: existingViewGroup.id },
id: viewGroup.id, { fieldValue: newOption.value },
},
{
fieldValue: newOption.value,
},
transactionManager, transactionManager,
); );
} }
@ -100,13 +96,49 @@ export class FieldMetadataRelatedRecordsService {
}, },
transactionManager, transactionManager,
); );
await this.syncNoValueViewGroup(
newFieldMetadata,
view,
viewGroupRepository,
transactionManager,
);
}
}
async syncNoValueViewGroup(
fieldMetadata: FieldMetadataEntity,
view: ViewWorkspaceEntity,
viewGroupRepository: WorkspaceRepository<ViewGroupWorkspaceEntity>,
transactionManager?: EntityManager,
): Promise<void> {
const noValueGroup = view.viewGroups.find(
(group) => group.fieldValue === '',
);
if (fieldMetadata.isNullable && !noValueGroup) {
const maxPosition = this.getMaxPosition(view.viewGroups);
const newGroup = viewGroupRepository.create({
fieldMetadataId: fieldMetadata.id,
fieldValue: '',
position: maxPosition + 1,
isVisible: true,
viewId: view.id,
});
await viewGroupRepository.insert(newGroup, transactionManager);
} else if (!fieldMetadata.isNullable && noValueGroup) {
await viewGroupRepository.delete(
{ id: noValueGroup.id },
transactionManager,
);
} }
} }
private getOptionsDifferences( private getOptionsDifferences(
oldOptions: (FieldMetadataDefaultOption | FieldMetadataComplexOption)[], oldOptions: (FieldMetadataDefaultOption | FieldMetadataComplexOption)[],
newOptions: (FieldMetadataDefaultOption | FieldMetadataComplexOption)[], newOptions: (FieldMetadataDefaultOption | FieldMetadataComplexOption)[],
) { ): Differences<FieldMetadataDefaultOption | FieldMetadataComplexOption> {
const differences: Differences< const differences: Differences<
FieldMetadataDefaultOption | FieldMetadataComplexOption FieldMetadataDefaultOption | FieldMetadataComplexOption
> = { > = {
@ -115,12 +147,8 @@ export class FieldMetadataRelatedRecordsService {
deleted: [], deleted: [],
}; };
const oldOptionsMap = new Map( const oldOptionsMap = new Map(oldOptions.map((opt) => [opt.id, opt]));
oldOptions.map((option) => [option.id, option]), const newOptionsMap = new Map(newOptions.map((opt) => [opt.id, opt]));
);
const newOptionsMap = new Map(
newOptions.map((option) => [option.id, option]),
);
for (const newOption of newOptions) { for (const newOption of newOptions) {
const oldOption = oldOptionsMap.get(newOption.id); const oldOption = oldOptionsMap.get(newOption.id);
@ -150,7 +178,7 @@ export class FieldMetadataRelatedRecordsService {
'view', 'view',
); );
return await viewRepository.find({ return viewRepository.find({
where: { where: {
viewGroups: { viewGroups: {
fieldMetadataId: fieldMetadata.id, fieldMetadataId: fieldMetadata.id,
@ -159,4 +187,8 @@ export class FieldMetadataRelatedRecordsService {
relations: ['viewGroups'], relations: ['viewGroups'],
}); });
} }
private getMaxPosition(viewGroups: ViewGroupWorkspaceEntity[]): number {
return viewGroups.reduce((max, group) => Math.max(max, group.position), 0);
}
} }