Simplifying a lot the upgrade system.
New way to upgrade:
`yarn command:prod upgrade`
New way to write upgrade commands (all wrapping is done for you)
```
override async runOnWorkspace({
index,
total,
workspaceId,
options,
}: RunOnWorkspaceArgs): Promise<void> {}
```
Also cleaning CommandModule imports to make it lighter
213 lines
7.2 KiB
TypeScript
213 lines
7.2 KiB
TypeScript
import { InjectRepository } from '@nestjs/typeorm';
|
|
|
|
import chalk from 'chalk';
|
|
import { Command } from 'nest-commander';
|
|
import { Repository } from 'typeorm';
|
|
import { v4 } from 'uuid';
|
|
|
|
import {
|
|
ActiveOrSuspendedWorkspacesMigrationCommandRunner,
|
|
RunOnWorkspaceArgs,
|
|
} from 'src/database/commands/command-runners/active-or-suspended-workspaces-migration.command-runner';
|
|
import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity';
|
|
import { FieldMetadataDefaultOption } from 'src/engine/metadata-modules/field-metadata/dtos/options.input';
|
|
import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity';
|
|
import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity';
|
|
import { WorkspaceMetadataVersionService } from 'src/engine/metadata-modules/workspace-metadata-version/services/workspace-metadata-version.service';
|
|
import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
|
|
import { tasksAssignedToMeView } from 'src/engine/workspace-manager/standard-objects-prefill-data/views/tasks-assigned-to-me';
|
|
import { TASK_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids';
|
|
import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids';
|
|
import { ViewFieldWorkspaceEntity } from 'src/modules/view/standard-objects/view-field.workspace-entity';
|
|
import { ViewFilterWorkspaceEntity } from 'src/modules/view/standard-objects/view-filter.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';
|
|
|
|
@Command({
|
|
name: 'upgrade:0-43:add-tasks-assigned-to-me-view',
|
|
description: 'Add tasks assigned to me view',
|
|
})
|
|
export class AddTasksAssignedToMeViewCommand extends ActiveOrSuspendedWorkspacesMigrationCommandRunner {
|
|
constructor(
|
|
@InjectRepository(Workspace, 'core')
|
|
protected readonly workspaceRepository: Repository<Workspace>,
|
|
@InjectRepository(ObjectMetadataEntity, 'metadata')
|
|
private readonly objectMetadataRepository: Repository<ObjectMetadataEntity>,
|
|
@InjectRepository(FieldMetadataEntity, 'metadata')
|
|
private readonly fieldMetadataRepository: Repository<FieldMetadataEntity>,
|
|
protected readonly twentyORMGlobalManager: TwentyORMGlobalManager,
|
|
private readonly workspaceMetadataVersionService: WorkspaceMetadataVersionService,
|
|
) {
|
|
super(workspaceRepository, twentyORMGlobalManager);
|
|
}
|
|
|
|
override async runOnWorkspace({
|
|
index,
|
|
total,
|
|
workspaceId,
|
|
}: RunOnWorkspaceArgs): Promise<void> {
|
|
this.logger.log(
|
|
`Running command for workspace ${workspaceId} ${index + 1}/${total}`,
|
|
);
|
|
|
|
await this.createTasksAssignedToMeView(workspaceId);
|
|
|
|
this.logger.log(
|
|
chalk.green(`Command completed for workspace ${workspaceId}.`),
|
|
);
|
|
}
|
|
|
|
private async createTasksAssignedToMeView(
|
|
workspaceId: string,
|
|
): Promise<void> {
|
|
const objectMetadata = await this.objectMetadataRepository.find({
|
|
where: { workspaceId },
|
|
relations: ['fields'],
|
|
});
|
|
|
|
const objectMetadataMap = objectMetadata.reduce((acc, object) => {
|
|
acc[object.standardId ?? ''] = {
|
|
id: object.id,
|
|
fields: object.fields.reduce((acc, field) => {
|
|
acc[field.standardId ?? ''] = field.id;
|
|
|
|
return acc;
|
|
}, {}),
|
|
};
|
|
|
|
return acc;
|
|
}, {});
|
|
|
|
const taskObjectMetadata = objectMetadata.find(
|
|
(object) => object.standardId === STANDARD_OBJECT_IDS.task,
|
|
);
|
|
|
|
if (!taskObjectMetadata) {
|
|
throw new Error(`Task object not found for workspace ${workspaceId}`);
|
|
}
|
|
|
|
const viewRepository =
|
|
await this.twentyORMGlobalManager.getRepositoryForWorkspace<ViewWorkspaceEntity>(
|
|
workspaceId,
|
|
'view',
|
|
false,
|
|
);
|
|
|
|
const existingView = await viewRepository.findOne({
|
|
where: {
|
|
name: 'Assigned to Me',
|
|
objectMetadataId: taskObjectMetadata.id,
|
|
},
|
|
});
|
|
|
|
if (existingView) {
|
|
this.logger.log(
|
|
chalk.yellow(
|
|
`"Assigned to Me" view already exists for workspace ${workspaceId}`,
|
|
),
|
|
);
|
|
|
|
return;
|
|
}
|
|
|
|
const viewDefinition = tasksAssignedToMeView(objectMetadataMap);
|
|
const viewId = v4();
|
|
|
|
const insertedView = await viewRepository.save({
|
|
id: viewId,
|
|
name: viewDefinition.name,
|
|
objectMetadataId: viewDefinition.objectMetadataId,
|
|
type: viewDefinition.type,
|
|
position: viewDefinition.position,
|
|
icon: viewDefinition.icon,
|
|
kanbanFieldMetadataId: viewDefinition.kanbanFieldMetadataId,
|
|
});
|
|
|
|
if (viewDefinition.fields && viewDefinition.fields.length > 0) {
|
|
const viewFieldRepository =
|
|
await this.twentyORMGlobalManager.getRepositoryForWorkspace<ViewFieldWorkspaceEntity>(
|
|
workspaceId,
|
|
'viewField',
|
|
false,
|
|
);
|
|
|
|
const viewFields = viewDefinition.fields.map((field) => ({
|
|
fieldMetadataId: field.fieldMetadataId,
|
|
position: field.position,
|
|
isVisible: field.isVisible,
|
|
size: field.size,
|
|
viewId: insertedView.id,
|
|
}));
|
|
|
|
await viewFieldRepository.save(viewFields);
|
|
}
|
|
|
|
if (viewDefinition.filters && viewDefinition.filters.length > 0) {
|
|
const viewFilterRepository =
|
|
await this.twentyORMGlobalManager.getRepositoryForWorkspace<ViewFilterWorkspaceEntity>(
|
|
workspaceId,
|
|
'viewFilter',
|
|
false,
|
|
);
|
|
|
|
const viewFilters = viewDefinition.filters.map((filter) => ({
|
|
fieldMetadataId: filter.fieldMetadataId,
|
|
displayValue: filter.displayValue,
|
|
operand: filter.operand,
|
|
value: filter.value,
|
|
viewId: insertedView.id,
|
|
}));
|
|
|
|
await viewFilterRepository.save(viewFilters);
|
|
}
|
|
|
|
await this.createTasksAssignedToMeViewGroups(workspaceId, insertedView.id);
|
|
}
|
|
|
|
private async createTasksAssignedToMeViewGroups(
|
|
workspaceId: string,
|
|
viewId: string,
|
|
) {
|
|
const taskStatusFieldMetadata = await this.fieldMetadataRepository.findOne({
|
|
where: {
|
|
workspaceId,
|
|
standardId: TASK_STANDARD_FIELD_IDS.status,
|
|
},
|
|
});
|
|
|
|
if (!taskStatusFieldMetadata) {
|
|
throw new Error(
|
|
`Task status field metadata not found for workspace ${workspaceId}`,
|
|
);
|
|
}
|
|
|
|
const optionValueViewGroups = taskStatusFieldMetadata.options.map(
|
|
(taskStatusOption: FieldMetadataDefaultOption, index) =>
|
|
({
|
|
fieldMetadataId: taskStatusFieldMetadata.id,
|
|
viewId,
|
|
fieldValue: taskStatusOption.value,
|
|
position: index,
|
|
}) satisfies Partial<ViewGroupWorkspaceEntity>,
|
|
);
|
|
|
|
const noValueViewGroup: Partial<ViewGroupWorkspaceEntity> = {
|
|
fieldMetadataId: taskStatusFieldMetadata.id,
|
|
viewId,
|
|
fieldValue: '',
|
|
position: optionValueViewGroups.length,
|
|
};
|
|
|
|
const viewGroups = [...optionValueViewGroups, noValueViewGroup];
|
|
|
|
const viewGroupRepository =
|
|
await this.twentyORMGlobalManager.getRepositoryForWorkspace<ViewGroupWorkspaceEntity>(
|
|
workspaceId,
|
|
'viewGroup',
|
|
false,
|
|
);
|
|
|
|
await viewGroupRepository.insert(viewGroups);
|
|
}
|
|
}
|