mirror of
https://github.com/Unleash/unleash.git
synced 2025-01-20 00:08:02 +01:00
30d8444c80
This PR cleans up and refactors the feature-strategy-store method getFeatureOverview to join on the new table and attempts to make the function more readable by extracting some of the logic into separate functions. Keeping the LastSeenMapper for now in case there is a reason to use it for the other endpoints.
1108 lines
33 KiB
TypeScript
1108 lines
33 KiB
TypeScript
import { subDays } from 'date-fns';
|
|
import { ValidationError } from 'joi';
|
|
import User, { IUser } from '../types/user';
|
|
import { AccessService, AccessWithRoles } from './access-service';
|
|
import NameExistsError from '../error/name-exists-error';
|
|
import InvalidOperationError from '../error/invalid-operation-error';
|
|
import { nameType } from '../routes/util';
|
|
import { projectSchema } from './project-schema';
|
|
import NotFoundError from '../error/notfound-error';
|
|
import {
|
|
DEFAULT_PROJECT,
|
|
FeatureToggle,
|
|
IAccountStore,
|
|
IEnvironmentStore,
|
|
IEventStore,
|
|
IFeatureEnvironmentStore,
|
|
IFeatureToggleStore,
|
|
IFeatureTypeStore,
|
|
IProject,
|
|
IProjectOverview,
|
|
IProjectWithCount,
|
|
IUnleashConfig,
|
|
IUnleashStores,
|
|
MOVE_FEATURE_TOGGLE,
|
|
PROJECT_CREATED,
|
|
PROJECT_DELETED,
|
|
PROJECT_UPDATED,
|
|
ProjectGroupAddedEvent,
|
|
ProjectGroupRemovedEvent,
|
|
ProjectGroupUpdateRoleEvent,
|
|
ProjectUserAddedEvent,
|
|
ProjectUserRemovedEvent,
|
|
ProjectUserUpdateRoleEvent,
|
|
RoleName,
|
|
IFlagResolver,
|
|
ProjectAccessAddedEvent,
|
|
ProjectAccessUserRolesUpdated,
|
|
ProjectAccessGroupRolesUpdated,
|
|
IProjectRoleUsage,
|
|
ProjectAccessUserRolesDeleted,
|
|
IFeatureNaming,
|
|
CreateProject,
|
|
} from '../types';
|
|
import {
|
|
IProjectQuery,
|
|
IProjectEnterpriseSettingsUpdate,
|
|
IProjectStore,
|
|
} from '../types/stores/project-store';
|
|
import {
|
|
IProjectAccessModel,
|
|
IRoleDescriptor,
|
|
} from '../types/stores/access-store';
|
|
import FeatureToggleService from './feature-toggle-service';
|
|
import IncompatibleProjectError from '../error/incompatible-project-error';
|
|
import ProjectWithoutOwnerError from '../error/project-without-owner-error';
|
|
import { arraysHaveSameItems } from '../util';
|
|
import { GroupService } from './group-service';
|
|
import { IGroupRole } from 'lib/types/group';
|
|
import { FavoritesService } from './favorites-service';
|
|
import { calculateAverageTimeToProd } from '../features/feature-toggle/time-to-production/time-to-production';
|
|
import { IProjectStatsStore } from 'lib/types/stores/project-stats-store-type';
|
|
import { uniqueByKey } from '../util/unique';
|
|
import { BadDataError, PermissionError } from '../error';
|
|
import { ProjectDoraMetricsSchema } from 'lib/openapi';
|
|
import { checkFeatureNamingData } from '../features/feature-naming-pattern/feature-naming-validation';
|
|
import { IPrivateProjectChecker } from '../features/private-project/privateProjectCheckerType';
|
|
import EventService from './event-service';
|
|
import { ILastSeenReadModel } from './client-metrics/last-seen/types/last-seen-read-model-type';
|
|
import { LastSeenMapper } from './client-metrics/last-seen/last-seen-mapper';
|
|
|
|
const getCreatedBy = (user: IUser) => user.email || user.username || 'unknown';
|
|
|
|
type Days = number;
|
|
type Count = number;
|
|
|
|
export interface IProjectStats {
|
|
avgTimeToProdCurrentWindow: Days;
|
|
createdCurrentWindow: Count;
|
|
createdPastWindow: Count;
|
|
archivedCurrentWindow: Count;
|
|
archivedPastWindow: Count;
|
|
projectActivityCurrentWindow: Count;
|
|
projectActivityPastWindow: Count;
|
|
projectMembersAddedCurrentWindow: Count;
|
|
}
|
|
|
|
interface ICalculateStatus {
|
|
projectId: string;
|
|
updates: IProjectStats;
|
|
}
|
|
|
|
export default class ProjectService {
|
|
private projectStore: IProjectStore;
|
|
|
|
private accessService: AccessService;
|
|
|
|
private eventStore: IEventStore;
|
|
|
|
private featureToggleStore: IFeatureToggleStore;
|
|
|
|
private featureTypeStore: IFeatureTypeStore;
|
|
|
|
private featureEnvironmentStore: IFeatureEnvironmentStore;
|
|
|
|
private environmentStore: IEnvironmentStore;
|
|
|
|
private groupService: GroupService;
|
|
|
|
private logger: any;
|
|
|
|
private featureToggleService: FeatureToggleService;
|
|
|
|
private privateProjectChecker: IPrivateProjectChecker;
|
|
|
|
private accountStore: IAccountStore;
|
|
|
|
private favoritesService: FavoritesService;
|
|
|
|
private eventService: EventService;
|
|
|
|
private projectStatsStore: IProjectStatsStore;
|
|
|
|
private lastSeenReadModel: ILastSeenReadModel;
|
|
|
|
private flagResolver: IFlagResolver;
|
|
|
|
private isEnterprise: boolean;
|
|
|
|
constructor(
|
|
{
|
|
projectStore,
|
|
eventStore,
|
|
featureToggleStore,
|
|
featureTypeStore,
|
|
environmentStore,
|
|
featureEnvironmentStore,
|
|
accountStore,
|
|
projectStatsStore,
|
|
}: Pick<
|
|
IUnleashStores,
|
|
| 'projectStore'
|
|
| 'eventStore'
|
|
| 'featureToggleStore'
|
|
| 'featureTypeStore'
|
|
| 'environmentStore'
|
|
| 'featureEnvironmentStore'
|
|
| 'accountStore'
|
|
| 'projectStatsStore'
|
|
>,
|
|
config: IUnleashConfig,
|
|
accessService: AccessService,
|
|
featureToggleService: FeatureToggleService,
|
|
groupService: GroupService,
|
|
favoriteService: FavoritesService,
|
|
eventService: EventService,
|
|
privateProjectChecker: IPrivateProjectChecker,
|
|
lastSeenReadModel: ILastSeenReadModel,
|
|
) {
|
|
this.projectStore = projectStore;
|
|
this.environmentStore = environmentStore;
|
|
this.featureEnvironmentStore = featureEnvironmentStore;
|
|
this.accessService = accessService;
|
|
this.eventStore = eventStore;
|
|
this.featureToggleStore = featureToggleStore;
|
|
this.featureTypeStore = featureTypeStore;
|
|
this.featureToggleService = featureToggleService;
|
|
this.favoritesService = favoriteService;
|
|
this.privateProjectChecker = privateProjectChecker;
|
|
this.accountStore = accountStore;
|
|
this.groupService = groupService;
|
|
this.eventService = eventService;
|
|
this.projectStatsStore = projectStatsStore;
|
|
this.lastSeenReadModel = lastSeenReadModel;
|
|
this.logger = config.getLogger('services/project-service.js');
|
|
this.flagResolver = config.flagResolver;
|
|
this.isEnterprise = config.isEnterprise;
|
|
}
|
|
|
|
async getProjects(
|
|
query?: IProjectQuery,
|
|
userId?: number,
|
|
): Promise<IProjectWithCount[]> {
|
|
const projects = await this.projectStore.getProjectsWithCounts(
|
|
query,
|
|
userId,
|
|
);
|
|
if (this.flagResolver.isEnabled('privateProjects') && userId) {
|
|
const projectAccess =
|
|
await this.privateProjectChecker.getUserAccessibleProjects(
|
|
userId,
|
|
);
|
|
|
|
if (projectAccess.mode === 'all') {
|
|
return projects;
|
|
} else {
|
|
return projects.filter((project) =>
|
|
projectAccess.projects.includes(project.id),
|
|
);
|
|
}
|
|
}
|
|
return projects;
|
|
}
|
|
|
|
async getProject(id: string): Promise<IProject> {
|
|
return this.projectStore.get(id);
|
|
}
|
|
|
|
private validateAndProcessFeatureNamingPattern = (
|
|
featureNaming: IFeatureNaming,
|
|
): IFeatureNaming => {
|
|
const validationResult = checkFeatureNamingData(featureNaming);
|
|
|
|
if (validationResult.state === 'invalid') {
|
|
const [firstReason, ...remainingReasons] =
|
|
validationResult.reasons.map((message) => ({
|
|
message,
|
|
}));
|
|
throw new BadDataError(
|
|
'The feature naming pattern data you provided was invalid.',
|
|
[firstReason, ...remainingReasons],
|
|
);
|
|
}
|
|
|
|
if (featureNaming.pattern && !featureNaming.example) {
|
|
featureNaming.example = null;
|
|
}
|
|
if (featureNaming.pattern && !featureNaming.description) {
|
|
featureNaming.description = null;
|
|
}
|
|
|
|
return featureNaming;
|
|
};
|
|
|
|
async createProject(
|
|
newProject: CreateProject,
|
|
user: IUser,
|
|
): Promise<IProject> {
|
|
const validatedData = await projectSchema.validateAsync(newProject);
|
|
const data = this.removeModeForNonEnterprise(validatedData);
|
|
await this.validateUniqueId(data.id);
|
|
|
|
await this.projectStore.create(data);
|
|
|
|
const enabledEnvironments = await this.environmentStore.getAll({
|
|
enabled: true,
|
|
});
|
|
|
|
// TODO: Only if enabled!
|
|
await Promise.all(
|
|
enabledEnvironments.map(async (e) => {
|
|
await this.featureEnvironmentStore.connectProject(
|
|
e.name,
|
|
data.id,
|
|
);
|
|
}),
|
|
);
|
|
|
|
await this.accessService.createDefaultProjectRoles(user, data.id);
|
|
|
|
await this.eventService.storeEvent({
|
|
type: PROJECT_CREATED,
|
|
createdBy: getCreatedBy(user),
|
|
data,
|
|
project: newProject.id,
|
|
});
|
|
|
|
return data;
|
|
}
|
|
|
|
async updateProject(updatedProject: IProject, user: User): Promise<void> {
|
|
const preData = await this.projectStore.get(updatedProject.id);
|
|
|
|
await this.projectStore.update(updatedProject);
|
|
|
|
await this.eventStore.store({
|
|
type: PROJECT_UPDATED,
|
|
project: updatedProject.id,
|
|
createdBy: getCreatedBy(user),
|
|
data: updatedProject,
|
|
preData,
|
|
});
|
|
}
|
|
|
|
async updateProjectEnterpriseSettings(
|
|
updatedProject: IProjectEnterpriseSettingsUpdate,
|
|
user: User,
|
|
): Promise<void> {
|
|
const preData = await this.projectStore.get(updatedProject.id);
|
|
|
|
if (updatedProject.featureNaming) {
|
|
this.validateAndProcessFeatureNamingPattern(
|
|
updatedProject.featureNaming,
|
|
);
|
|
}
|
|
|
|
await this.projectStore.updateProjectEnterpriseSettings(updatedProject);
|
|
|
|
await this.eventService.storeEvent({
|
|
type: PROJECT_UPDATED,
|
|
project: updatedProject.id,
|
|
createdBy: getCreatedBy(user),
|
|
data: updatedProject,
|
|
preData,
|
|
});
|
|
}
|
|
|
|
async checkProjectsCompatibility(
|
|
feature: FeatureToggle,
|
|
newProjectId: string,
|
|
): Promise<boolean> {
|
|
const featureEnvs = await this.featureEnvironmentStore.getAll({
|
|
feature_name: feature.name,
|
|
});
|
|
const newEnvs = await this.projectStore.getEnvironmentsForProject(
|
|
newProjectId,
|
|
);
|
|
return arraysHaveSameItems(
|
|
featureEnvs.map((env) => env.environment),
|
|
newEnvs.map((projectEnv) => projectEnv.environment),
|
|
);
|
|
}
|
|
|
|
async addEnvironmentToProject(
|
|
project: string,
|
|
environment: string,
|
|
): Promise<void> {
|
|
await this.projectStore.addEnvironmentToProject(project, environment);
|
|
}
|
|
|
|
async changeProject(
|
|
newProjectId: string,
|
|
featureName: string,
|
|
user: User,
|
|
currentProjectId: string,
|
|
): Promise<any> {
|
|
const feature = await this.featureToggleStore.get(featureName);
|
|
|
|
if (feature.project !== currentProjectId) {
|
|
throw new PermissionError(MOVE_FEATURE_TOGGLE);
|
|
}
|
|
const project = await this.getProject(newProjectId);
|
|
|
|
if (!project) {
|
|
throw new NotFoundError(`Project ${newProjectId} not found`);
|
|
}
|
|
|
|
const authorized = await this.accessService.hasPermission(
|
|
user,
|
|
MOVE_FEATURE_TOGGLE,
|
|
newProjectId,
|
|
);
|
|
|
|
if (!authorized) {
|
|
throw new PermissionError(MOVE_FEATURE_TOGGLE);
|
|
}
|
|
|
|
const isCompatibleWithTargetProject =
|
|
await this.checkProjectsCompatibility(feature, newProjectId);
|
|
if (!isCompatibleWithTargetProject) {
|
|
throw new IncompatibleProjectError(newProjectId);
|
|
}
|
|
const updatedFeature = await this.featureToggleService.changeProject(
|
|
featureName,
|
|
newProjectId,
|
|
getCreatedBy(user),
|
|
);
|
|
await this.featureToggleService.updateFeatureStrategyProject(
|
|
featureName,
|
|
newProjectId,
|
|
);
|
|
|
|
return updatedFeature;
|
|
}
|
|
|
|
async deleteProject(id: string, user: User): Promise<void> {
|
|
if (id === DEFAULT_PROJECT) {
|
|
throw new InvalidOperationError(
|
|
'You can not delete the default project!',
|
|
);
|
|
}
|
|
|
|
const toggles = await this.featureToggleStore.getAll({
|
|
project: id,
|
|
archived: false,
|
|
});
|
|
|
|
if (toggles.length > 0) {
|
|
throw new InvalidOperationError(
|
|
'You can not delete a project with active feature toggles',
|
|
);
|
|
}
|
|
|
|
await this.projectStore.delete(id);
|
|
|
|
await this.eventService.storeEvent({
|
|
type: PROJECT_DELETED,
|
|
createdBy: getCreatedBy(user),
|
|
project: id,
|
|
});
|
|
|
|
await this.accessService.removeDefaultProjectRoles(user, id);
|
|
}
|
|
|
|
async validateId(id: string): Promise<boolean> {
|
|
await nameType.validateAsync(id);
|
|
await this.validateUniqueId(id);
|
|
return true;
|
|
}
|
|
|
|
async validateUniqueId(id: string): Promise<void> {
|
|
const exists = await this.projectStore.hasProject(id);
|
|
if (exists) {
|
|
throw new NameExistsError('A project with this id already exists.');
|
|
}
|
|
}
|
|
|
|
// RBAC methods
|
|
async getAccessToProject(projectId: string): Promise<AccessWithRoles> {
|
|
return this.accessService.getProjectRoleAccess(projectId);
|
|
}
|
|
|
|
// Deprecated: See addAccess instead.
|
|
async addUser(
|
|
projectId: string,
|
|
roleId: number,
|
|
userId: number,
|
|
createdBy: string,
|
|
): Promise<void> {
|
|
const { roles, users } = await this.accessService.getProjectRoleAccess(
|
|
projectId,
|
|
);
|
|
const user = await this.accountStore.get(userId);
|
|
|
|
const role = roles.find((r) => r.id === roleId);
|
|
if (!role) {
|
|
throw new NotFoundError(
|
|
`Could not find roleId=${roleId} on project=${projectId}`,
|
|
);
|
|
}
|
|
|
|
const alreadyHasAccess = users.some((u) => u.id === userId);
|
|
if (alreadyHasAccess) {
|
|
throw new Error(`User already has access to project=${projectId}`);
|
|
}
|
|
|
|
await this.accessService.addUserToRole(userId, role.id, projectId);
|
|
|
|
await this.eventService.storeEvent(
|
|
new ProjectUserAddedEvent({
|
|
project: projectId,
|
|
createdBy: createdBy || 'system-user',
|
|
data: {
|
|
roleId,
|
|
userId,
|
|
roleName: role.name,
|
|
email: user.email,
|
|
},
|
|
}),
|
|
);
|
|
}
|
|
|
|
async removeUser(
|
|
projectId: string,
|
|
roleId: number,
|
|
userId: number,
|
|
createdBy: string,
|
|
): Promise<void> {
|
|
const role = await this.findProjectRole(projectId, roleId);
|
|
|
|
await this.validateAtLeastOneOwner(projectId, role);
|
|
|
|
await this.accessService.removeUserFromRole(userId, role.id, projectId);
|
|
|
|
const user = await this.accountStore.get(userId);
|
|
|
|
await this.eventService.storeEvent(
|
|
new ProjectUserRemovedEvent({
|
|
project: projectId,
|
|
createdBy,
|
|
preData: {
|
|
roleId,
|
|
userId,
|
|
roleName: role.name,
|
|
email: user.email,
|
|
},
|
|
}),
|
|
);
|
|
}
|
|
|
|
async removeUserAccess(
|
|
projectId: string,
|
|
userId: number,
|
|
createdBy: string,
|
|
): Promise<void> {
|
|
const existingRoles = await this.accessService.getProjectRolesForUser(
|
|
projectId,
|
|
userId,
|
|
);
|
|
|
|
await this.accessService.removeUserAccess(projectId, userId);
|
|
|
|
await this.eventService.storeEvent(
|
|
new ProjectAccessUserRolesDeleted({
|
|
project: projectId,
|
|
createdBy,
|
|
preData: {
|
|
roles: existingRoles,
|
|
userId,
|
|
},
|
|
}),
|
|
);
|
|
}
|
|
|
|
async removeGroupAccess(
|
|
projectId: string,
|
|
groupId: number,
|
|
createdBy: string,
|
|
): Promise<void> {
|
|
const existingRoles = await this.accessService.getProjectRolesForGroup(
|
|
projectId,
|
|
groupId,
|
|
);
|
|
|
|
await this.accessService.removeGroupAccess(projectId, groupId);
|
|
|
|
await this.eventService.storeEvent(
|
|
new ProjectAccessUserRolesDeleted({
|
|
project: projectId,
|
|
createdBy,
|
|
preData: {
|
|
roles: existingRoles,
|
|
groupId,
|
|
},
|
|
}),
|
|
);
|
|
}
|
|
|
|
async addGroup(
|
|
projectId: string,
|
|
roleId: number,
|
|
groupId: number,
|
|
modifiedBy: string,
|
|
): Promise<void> {
|
|
const role = await this.accessService.getRole(roleId);
|
|
const group = await this.groupService.getGroup(groupId);
|
|
const project = await this.getProject(projectId);
|
|
if (group.id == null)
|
|
throw new ValidationError(
|
|
'Unexpected empty group id',
|
|
[],
|
|
undefined,
|
|
);
|
|
|
|
await this.accessService.addGroupToRole(
|
|
group.id,
|
|
role.id,
|
|
modifiedBy,
|
|
project.id,
|
|
);
|
|
|
|
await this.eventService.storeEvent(
|
|
new ProjectGroupAddedEvent({
|
|
project: project.id,
|
|
createdBy: modifiedBy,
|
|
data: {
|
|
groupId: group.id,
|
|
projectId: project.id,
|
|
roleName: role.name,
|
|
},
|
|
}),
|
|
);
|
|
}
|
|
|
|
async removeGroup(
|
|
projectId: string,
|
|
roleId: number,
|
|
groupId: number,
|
|
modifiedBy: string,
|
|
): Promise<void> {
|
|
const group = await this.groupService.getGroup(groupId);
|
|
const role = await this.accessService.getRole(roleId);
|
|
const project = await this.getProject(projectId);
|
|
if (group.id == null)
|
|
throw new ValidationError(
|
|
'Unexpected empty group id',
|
|
[],
|
|
undefined,
|
|
);
|
|
|
|
await this.accessService.removeGroupFromRole(
|
|
group.id,
|
|
role.id,
|
|
project.id,
|
|
);
|
|
|
|
await this.eventService.storeEvent(
|
|
new ProjectGroupRemovedEvent({
|
|
project: projectId,
|
|
createdBy: modifiedBy,
|
|
preData: {
|
|
groupId: group.id,
|
|
projectId: project.id,
|
|
roleName: role.name,
|
|
},
|
|
}),
|
|
);
|
|
}
|
|
|
|
async addRoleAccess(
|
|
projectId: string,
|
|
roleId: number,
|
|
usersAndGroups: IProjectAccessModel,
|
|
createdBy: string,
|
|
): Promise<void> {
|
|
await this.accessService.addRoleAccessToProject(
|
|
usersAndGroups.users,
|
|
usersAndGroups.groups,
|
|
projectId,
|
|
roleId,
|
|
createdBy,
|
|
);
|
|
|
|
await this.eventService.storeEvent(
|
|
new ProjectAccessAddedEvent({
|
|
project: projectId,
|
|
createdBy,
|
|
data: {
|
|
roleId,
|
|
groups: usersAndGroups.groups.map(({ id }) => id),
|
|
users: usersAndGroups.users.map(({ id }) => id),
|
|
},
|
|
}),
|
|
);
|
|
}
|
|
|
|
async addAccess(
|
|
projectId: string,
|
|
roles: number[],
|
|
groups: number[],
|
|
users: number[],
|
|
createdBy: string,
|
|
): Promise<void> {
|
|
await this.accessService.addAccessToProject(
|
|
roles,
|
|
groups,
|
|
users,
|
|
projectId,
|
|
createdBy,
|
|
);
|
|
|
|
await this.eventService.storeEvent(
|
|
new ProjectAccessAddedEvent({
|
|
project: projectId,
|
|
createdBy,
|
|
data: {
|
|
roles,
|
|
groups,
|
|
users,
|
|
},
|
|
}),
|
|
);
|
|
}
|
|
|
|
async setRolesForUser(
|
|
projectId: string,
|
|
userId: number,
|
|
roles: number[],
|
|
createdByUserName: string,
|
|
): Promise<void> {
|
|
const existingRoles = await this.accessService.getProjectRolesForUser(
|
|
projectId,
|
|
userId,
|
|
);
|
|
await this.accessService.setProjectRolesForUser(
|
|
projectId,
|
|
userId,
|
|
roles,
|
|
);
|
|
await this.eventService.storeEvent(
|
|
new ProjectAccessUserRolesUpdated({
|
|
project: projectId,
|
|
createdBy: createdByUserName,
|
|
data: {
|
|
roles,
|
|
userId,
|
|
},
|
|
preData: {
|
|
roles: existingRoles,
|
|
userId,
|
|
},
|
|
}),
|
|
);
|
|
}
|
|
|
|
async setRolesForGroup(
|
|
projectId: string,
|
|
groupId: number,
|
|
roles: number[],
|
|
createdBy: string,
|
|
): Promise<void> {
|
|
const existingRoles = await this.accessService.getProjectRolesForGroup(
|
|
projectId,
|
|
groupId,
|
|
);
|
|
await this.accessService.setProjectRolesForGroup(
|
|
projectId,
|
|
groupId,
|
|
roles,
|
|
createdBy,
|
|
);
|
|
await this.eventService.storeEvent(
|
|
new ProjectAccessGroupRolesUpdated({
|
|
project: projectId,
|
|
createdBy,
|
|
data: {
|
|
roles,
|
|
groupId,
|
|
},
|
|
preData: {
|
|
roles: existingRoles,
|
|
groupId,
|
|
},
|
|
}),
|
|
);
|
|
}
|
|
|
|
async findProjectGroupRole(
|
|
projectId: string,
|
|
roleId: number,
|
|
): Promise<IGroupRole> {
|
|
const roles = await this.groupService.getRolesForProject(projectId);
|
|
const role = roles.find((r) => r.roleId === roleId);
|
|
if (!role) {
|
|
throw new NotFoundError(
|
|
`Couldn't find roleId=${roleId} on project=${projectId}`,
|
|
);
|
|
}
|
|
return role;
|
|
}
|
|
|
|
async findProjectRole(
|
|
projectId: string,
|
|
roleId: number,
|
|
): Promise<IRoleDescriptor> {
|
|
const roles = await this.accessService.getRolesForProject(projectId);
|
|
const role = roles.find((r) => r.id === roleId);
|
|
if (!role) {
|
|
throw new NotFoundError(
|
|
`Couldn't find roleId=${roleId} on project=${projectId}`,
|
|
);
|
|
}
|
|
return role;
|
|
}
|
|
|
|
async validateAtLeastOneOwner(
|
|
projectId: string,
|
|
currentRole: IRoleDescriptor,
|
|
): Promise<void> {
|
|
if (currentRole.name === RoleName.OWNER) {
|
|
const users = await this.accessService.getProjectUsersForRole(
|
|
currentRole.id,
|
|
projectId,
|
|
);
|
|
const groups = await this.groupService.getProjectGroups(projectId);
|
|
const roleGroups = groups.filter(
|
|
(g) => g.roleId === currentRole.id,
|
|
);
|
|
if (users.length + roleGroups.length < 2) {
|
|
throw new ProjectWithoutOwnerError();
|
|
}
|
|
}
|
|
}
|
|
|
|
async getDoraMetrics(projectId: string): Promise<ProjectDoraMetricsSchema> {
|
|
const activeFeatureToggles = (
|
|
await this.featureToggleStore.getAll({ project: projectId })
|
|
).map((feature) => feature.name);
|
|
|
|
const archivedFeatureToggles = (
|
|
await this.featureToggleStore.getAll({
|
|
project: projectId,
|
|
archived: true,
|
|
})
|
|
).map((feature) => feature.name);
|
|
|
|
const featureToggleNames = [
|
|
...activeFeatureToggles,
|
|
...archivedFeatureToggles,
|
|
];
|
|
|
|
const projectAverage = calculateAverageTimeToProd(
|
|
await this.projectStatsStore.getTimeToProdDates(projectId),
|
|
);
|
|
|
|
const toggleAverage =
|
|
await this.projectStatsStore.getTimeToProdDatesForFeatureToggles(
|
|
projectId,
|
|
featureToggleNames,
|
|
);
|
|
|
|
return { features: toggleAverage, projectAverage: projectAverage };
|
|
}
|
|
|
|
async changeRole(
|
|
projectId: string,
|
|
roleId: number,
|
|
userId: number,
|
|
createdBy: string,
|
|
): Promise<void> {
|
|
const usersWithRoles = await this.getAccessToProject(projectId);
|
|
const user = usersWithRoles.users.find((u) => u.id === userId);
|
|
if (!user)
|
|
throw new ValidationError('Unexpected empty user', [], undefined);
|
|
|
|
const currentRole = usersWithRoles.roles.find(
|
|
(r) => r.id === user.roleId,
|
|
);
|
|
if (!currentRole)
|
|
throw new ValidationError(
|
|
'Unexpected empty current role',
|
|
[],
|
|
undefined,
|
|
);
|
|
|
|
if (currentRole.id === roleId) {
|
|
// Nothing to do....
|
|
return;
|
|
}
|
|
|
|
await this.validateAtLeastOneOwner(projectId, currentRole);
|
|
|
|
await this.accessService.updateUserProjectRole(
|
|
userId,
|
|
roleId,
|
|
projectId,
|
|
);
|
|
const role = await this.findProjectRole(projectId, roleId);
|
|
|
|
await this.eventService.storeEvent(
|
|
new ProjectUserUpdateRoleEvent({
|
|
project: projectId,
|
|
createdBy,
|
|
preData: {
|
|
userId,
|
|
roleId: currentRole.id,
|
|
roleName: currentRole.name,
|
|
email: user.email,
|
|
},
|
|
data: {
|
|
userId,
|
|
roleId,
|
|
roleName: role.name,
|
|
email: user.email,
|
|
},
|
|
}),
|
|
);
|
|
}
|
|
|
|
async changeGroupRole(
|
|
projectId: string,
|
|
roleId: number,
|
|
userId: number,
|
|
createdBy: string,
|
|
): Promise<void> {
|
|
const usersWithRoles = await this.getAccessToProject(projectId);
|
|
const user = usersWithRoles.groups.find((u) => u.id === userId);
|
|
if (!user)
|
|
throw new ValidationError('Unexpected empty user', [], undefined);
|
|
const currentRole = usersWithRoles.roles.find(
|
|
(r) => r.id === user.roleId,
|
|
);
|
|
if (!currentRole)
|
|
throw new ValidationError(
|
|
'Unexpected empty current role',
|
|
[],
|
|
undefined,
|
|
);
|
|
|
|
if (currentRole.id === roleId) {
|
|
// Nothing to do....
|
|
return;
|
|
}
|
|
|
|
await this.validateAtLeastOneOwner(projectId, currentRole);
|
|
|
|
await this.accessService.updateGroupProjectRole(
|
|
userId,
|
|
roleId,
|
|
projectId,
|
|
);
|
|
const role = await this.findProjectGroupRole(projectId, roleId);
|
|
|
|
await this.eventService.storeEvent(
|
|
new ProjectGroupUpdateRoleEvent({
|
|
project: projectId,
|
|
createdBy,
|
|
preData: {
|
|
userId,
|
|
roleId: currentRole.id,
|
|
roleName: currentRole.name,
|
|
},
|
|
data: {
|
|
userId,
|
|
roleId,
|
|
roleName: role.name,
|
|
},
|
|
}),
|
|
);
|
|
}
|
|
|
|
async getMembers(projectId: string): Promise<number> {
|
|
return this.projectStore.getMembersCountByProject(projectId);
|
|
}
|
|
|
|
async getProjectUsers(
|
|
projectId: string,
|
|
): Promise<Array<Pick<IUser, 'id' | 'email' | 'username'>>> {
|
|
const { groups, users } = await this.accessService.getProjectRoleAccess(
|
|
projectId,
|
|
);
|
|
const actualUsers = users.map((user) => ({
|
|
id: user.id,
|
|
email: user.email,
|
|
username: user.username,
|
|
}));
|
|
const actualGroupUsers = groups
|
|
.flatMap((group) => group.users)
|
|
.map((user) => user.user)
|
|
.map((user) => ({
|
|
id: user.id,
|
|
email: user.email,
|
|
username: user.username,
|
|
}));
|
|
return uniqueByKey([...actualUsers, ...actualGroupUsers], 'id');
|
|
}
|
|
|
|
async isProjectUser(userId: number, projectId: string): Promise<boolean> {
|
|
const users = await this.getProjectUsers(projectId);
|
|
return Boolean(users.find((user) => user.id === userId));
|
|
}
|
|
|
|
async getProjectsByUser(userId: number): Promise<string[]> {
|
|
return this.projectStore.getProjectsByUser(userId);
|
|
}
|
|
|
|
async getProjectRoleUsage(roleId: number): Promise<IProjectRoleUsage[]> {
|
|
return this.accessService.getProjectRoleUsage(roleId);
|
|
}
|
|
|
|
async statusJob(): Promise<void> {
|
|
const projects = await this.projectStore.getAll();
|
|
|
|
const statusUpdates = await Promise.all(
|
|
projects.map((project) => this.getStatusUpdates(project.id)),
|
|
);
|
|
|
|
await Promise.all(
|
|
statusUpdates.map((statusUpdate) => {
|
|
return this.projectStatsStore.updateProjectStats(
|
|
statusUpdate.projectId,
|
|
statusUpdate.updates,
|
|
);
|
|
}),
|
|
);
|
|
}
|
|
|
|
async getStatusUpdates(projectId: string): Promise<ICalculateStatus> {
|
|
const dateMinusThirtyDays = subDays(new Date(), 30).toISOString();
|
|
const dateMinusSixtyDays = subDays(new Date(), 60).toISOString();
|
|
|
|
const [
|
|
createdCurrentWindow,
|
|
createdPastWindow,
|
|
archivedCurrentWindow,
|
|
archivedPastWindow,
|
|
] = await Promise.all([
|
|
await this.featureToggleStore.countByDate({
|
|
project: projectId,
|
|
dateAccessor: 'created_at',
|
|
date: dateMinusThirtyDays,
|
|
}),
|
|
await this.featureToggleStore.countByDate({
|
|
project: projectId,
|
|
dateAccessor: 'created_at',
|
|
range: [dateMinusSixtyDays, dateMinusThirtyDays],
|
|
}),
|
|
await this.featureToggleStore.countByDate({
|
|
project: projectId,
|
|
archived: true,
|
|
dateAccessor: 'archived_at',
|
|
date: dateMinusThirtyDays,
|
|
}),
|
|
await this.featureToggleStore.countByDate({
|
|
project: projectId,
|
|
archived: true,
|
|
dateAccessor: 'archived_at',
|
|
range: [dateMinusSixtyDays, dateMinusThirtyDays],
|
|
}),
|
|
]);
|
|
|
|
const [projectActivityCurrentWindow, projectActivityPastWindow] =
|
|
await Promise.all([
|
|
this.eventStore.queryCount([
|
|
{ op: 'where', parameters: { project: projectId } },
|
|
{
|
|
op: 'beforeDate',
|
|
parameters: {
|
|
dateAccessor: 'created_at',
|
|
date: dateMinusThirtyDays,
|
|
},
|
|
},
|
|
]),
|
|
this.eventStore.queryCount([
|
|
{ op: 'where', parameters: { project: projectId } },
|
|
{
|
|
op: 'betweenDate',
|
|
parameters: {
|
|
dateAccessor: 'created_at',
|
|
range: [dateMinusSixtyDays, dateMinusThirtyDays],
|
|
},
|
|
},
|
|
]),
|
|
]);
|
|
|
|
const avgTimeToProdCurrentWindow = calculateAverageTimeToProd(
|
|
await this.projectStatsStore.getTimeToProdDates(projectId),
|
|
);
|
|
|
|
const projectMembersAddedCurrentWindow =
|
|
await this.projectStore.getMembersCountByProjectAfterDate(
|
|
projectId,
|
|
dateMinusThirtyDays,
|
|
);
|
|
|
|
return {
|
|
projectId,
|
|
updates: {
|
|
avgTimeToProdCurrentWindow,
|
|
createdCurrentWindow,
|
|
createdPastWindow,
|
|
archivedCurrentWindow,
|
|
archivedPastWindow,
|
|
projectActivityCurrentWindow,
|
|
projectActivityPastWindow,
|
|
projectMembersAddedCurrentWindow,
|
|
},
|
|
};
|
|
}
|
|
|
|
async getProjectOverview(
|
|
projectId: string,
|
|
archived: boolean = false,
|
|
userId?: number,
|
|
): Promise<IProjectOverview> {
|
|
const [
|
|
project,
|
|
environments,
|
|
features,
|
|
members,
|
|
favorite,
|
|
projectStats,
|
|
] = await Promise.all([
|
|
this.projectStore.get(projectId),
|
|
this.projectStore.getEnvironmentsForProject(projectId),
|
|
this.featureToggleService.getFeatureOverview({
|
|
projectId,
|
|
archived,
|
|
userId,
|
|
}),
|
|
this.projectStore.getMembersCountByProject(projectId),
|
|
userId
|
|
? this.favoritesService.isFavoriteProject({
|
|
project: projectId,
|
|
userId,
|
|
})
|
|
: Promise.resolve(false),
|
|
this.projectStatsStore.getProjectStats(projectId),
|
|
]);
|
|
|
|
return {
|
|
stats: projectStats,
|
|
name: project.name,
|
|
description: project.description,
|
|
mode: project.mode,
|
|
featureLimit: project.featureLimit,
|
|
featureNaming: project.featureNaming,
|
|
defaultStickiness: project.defaultStickiness,
|
|
health: project.health || 0,
|
|
favorite: favorite,
|
|
updatedAt: project.updatedAt,
|
|
createdAt: project.createdAt,
|
|
environments,
|
|
features: features,
|
|
members,
|
|
version: 1,
|
|
};
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types
|
|
removeModeForNonEnterprise(data): any {
|
|
if (this.isEnterprise) {
|
|
return data;
|
|
}
|
|
const { mode, ...proData } = data;
|
|
return proData;
|
|
}
|
|
}
|