123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210 |
- import {createStore} from 'reflux';
- import ProjectActions from 'sentry/actions/projectActions';
- import {Project, Team} from 'sentry/types';
- import {makeSafeRefluxStore} from 'sentry/utils/makeSafeRefluxStore';
- import {CommonStoreDefinition} from './types';
- type State = {
- loading: boolean;
- projects: Project[];
- };
- type StatsData = Record<string, Project['stats']>;
- /**
- * Attributes that need typing but aren't part of the external interface,
- */
- type InternalDefinition = {
- itemsById: Record<string, Project>;
- loading: boolean;
- removeTeamFromProject(teamSlug: string, project: Project): void;
- };
- interface ProjectsStoreDefinition
- extends InternalDefinition,
- CommonStoreDefinition<State> {
- getAll(): Project[];
- getById(id?: string): Project | undefined;
- getBySlug(slug?: string): Project | undefined;
- init(): void;
- isLoading(): boolean;
- loadInitialData(projects: Project[]): void;
- onAddTeam(team: Team, projectSlug: string): void;
- onChangeSlug(prevSlug: string, newSlug: string): void;
- onCreateSuccess(project: Project): void;
- onDeleteTeam(slug: string): void;
- onRemoveTeam(teamSlug: string, projectSlug: string): void;
- onStatsLoadSuccess(data: StatsData): void;
- onUpdateSuccess(data: Partial<Project>): void;
- reset(): void;
- }
- const storeConfig: ProjectsStoreDefinition = {
- itemsById: {},
- loading: true,
- unsubscribeListeners: [],
- init() {
- this.reset();
- this.unsubscribeListeners.push(
- this.listenTo(ProjectActions.addTeamSuccess, this.onAddTeam)
- );
- this.unsubscribeListeners.push(
- this.listenTo(ProjectActions.changeSlug, this.onChangeSlug)
- );
- this.unsubscribeListeners.push(
- this.listenTo(ProjectActions.createSuccess, this.onCreateSuccess)
- );
- this.unsubscribeListeners.push(
- this.listenTo(ProjectActions.loadProjects, this.loadInitialData)
- );
- this.unsubscribeListeners.push(
- this.listenTo(ProjectActions.loadStatsSuccess, this.onStatsLoadSuccess)
- );
- this.unsubscribeListeners.push(
- this.listenTo(ProjectActions.removeTeamSuccess, this.onRemoveTeam)
- );
- this.unsubscribeListeners.push(this.listenTo(ProjectActions.reset, this.reset));
- this.unsubscribeListeners.push(
- this.listenTo(ProjectActions.updateSuccess, this.onUpdateSuccess)
- );
- },
- reset() {
- this.itemsById = {};
- this.loading = true;
- },
- loadInitialData(items: Project[]) {
- const mapping = items.map(project => [project.id, project] as const);
- this.itemsById = Object.fromEntries(mapping);
- this.loading = false;
- this.trigger(new Set(Object.keys(this.itemsById)));
- },
- onChangeSlug(prevSlug: string, newSlug: string) {
- const prevProject = this.getBySlug(prevSlug);
- if (!prevProject) {
- return;
- }
- const newProject = {...prevProject, slug: newSlug};
- this.itemsById = {...this.itemsById, [newProject.id]: newProject};
- this.trigger(new Set([prevProject.id]));
- },
- onCreateSuccess(project: Project) {
- this.itemsById = {...this.itemsById, [project.id]: project};
- this.trigger(new Set([project.id]));
- },
- onUpdateSuccess(data: Partial<Project>) {
- const project = this.getById(data.id);
- if (!project) {
- return;
- }
- const newProject = {...project, ...data};
- this.itemsById = {...this.itemsById, [project.id]: newProject};
- this.trigger(new Set([data.id]));
- },
- onStatsLoadSuccess(data) {
- const entries = Object.entries(data || {}).filter(
- ([projectId]) => projectId in this.itemsById
- );
- // Assign stats into projects
- entries.forEach(([projectId, stats]) => {
- this.itemsById[projectId].stats = stats;
- });
- const touchedIds = entries.map(([projectId]) => projectId);
- this.trigger(new Set(touchedIds));
- },
- /**
- * Listener for when a team is completely removed
- *
- * @param teamSlug Team Slug
- */
- onDeleteTeam(teamSlug: string) {
- // Look for team in all projects
- const projects = this.getAll().filter(({teams}) =>
- teams.find(({slug}) => slug === teamSlug)
- );
- projects.forEach(project => this.removeTeamFromProject(teamSlug, project));
- const affectedProjectIds = projects.map(project => project.id);
- this.trigger(new Set(affectedProjectIds));
- },
- onRemoveTeam(teamSlug: string, projectSlug: string) {
- const project = this.getBySlug(projectSlug);
- if (!project) {
- return;
- }
- this.removeTeamFromProject(teamSlug, project);
- this.trigger(new Set([project.id]));
- },
- onAddTeam(team: Team, projectSlug: string) {
- const project = this.getBySlug(projectSlug);
- // Don't do anything if we can't find a project
- if (!project) {
- return;
- }
- const newProject = {...project, teams: [...project.teams, team]};
- this.itemsById = {...this.itemsById, [project.id]: newProject};
- this.trigger(new Set([project.id]));
- },
- // Internal method, does not trigger
- removeTeamFromProject(teamSlug: string, project: Project) {
- const newTeams = project.teams.filter(({slug}) => slug !== teamSlug);
- const newProject = {...project, teams: newTeams};
- this.itemsById = {...this.itemsById, [project.id]: newProject};
- },
- isLoading() {
- return this.loading;
- },
- getAll() {
- return Object.values(this.itemsById).sort((a, b) => a.slug.localeCompare(b.slug));
- },
- getById(id) {
- return this.getAll().find(project => project.id === id);
- },
- getBySlug(slug) {
- return this.getAll().find(project => project.slug === slug);
- },
- getState() {
- return {
- projects: this.getAll(),
- loading: this.loading,
- };
- },
- };
- const ProjectsStore = createStore(makeSafeRefluxStore(storeConfig));
- export default ProjectsStore;
|