organizationStore.tsx 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  1. import Reflux from 'reflux';
  2. import OrganizationActions from 'app/actions/organizationActions';
  3. import ProjectActions from 'app/actions/projectActions';
  4. import TeamActions from 'app/actions/teamActions';
  5. import {ORGANIZATION_FETCH_ERROR_TYPES} from 'app/constants';
  6. import {Organization, Project, Team} from 'app/types';
  7. import RequestError from 'app/utils/requestError/requestError';
  8. type UpdateOptions = {
  9. replace?: boolean;
  10. };
  11. type OutputState = {
  12. organization: Organization | null;
  13. loading: boolean;
  14. dirty: boolean;
  15. errorType?: string | null;
  16. error?: RequestError | null;
  17. };
  18. type OrganizationStoreInterface = {
  19. init: () => void;
  20. reset: () => void;
  21. onUpdate: (org: Organization, options: UpdateOptions) => void;
  22. onFetchOrgError: (err: RequestError) => void;
  23. onProjectOrTeamChange: () => void;
  24. onLoadProjects: (projects: Project[]) => void;
  25. onLoadTeams: (teams: Team[]) => void;
  26. get: () => OutputState;
  27. };
  28. const storeConfig: Reflux.StoreDefinition & OrganizationStoreInterface = {
  29. init() {
  30. this.reset();
  31. this.listenTo(OrganizationActions.update, this.onUpdate);
  32. this.listenTo(OrganizationActions.fetchOrg, this.reset);
  33. this.listenTo(OrganizationActions.fetchOrgError, this.onFetchOrgError);
  34. // fill in teams and projects if they are loaded
  35. this.listenTo(ProjectActions.loadProjects, this.onLoadProjects);
  36. this.listenTo(TeamActions.loadTeams, this.onLoadTeams);
  37. // mark the store as dirty if projects or teams change
  38. this.listenTo(ProjectActions.createSuccess, this.onProjectOrTeamChange);
  39. this.listenTo(ProjectActions.updateSuccess, this.onProjectOrTeamChange);
  40. this.listenTo(ProjectActions.changeSlug, this.onProjectOrTeamChange);
  41. this.listenTo(ProjectActions.addTeamSuccess, this.onProjectOrTeamChange);
  42. this.listenTo(ProjectActions.removeTeamSuccess, this.onProjectOrTeamChange);
  43. this.listenTo(TeamActions.updateSuccess, this.onProjectOrTeamChange);
  44. this.listenTo(TeamActions.removeTeamSuccess, this.onProjectOrTeamChange);
  45. this.listenTo(TeamActions.createTeamSuccess, this.onProjectOrTeamChange);
  46. },
  47. reset() {
  48. this.loading = true;
  49. this.error = null;
  50. this.errorType = null;
  51. this.organization = null;
  52. this.dirty = false;
  53. this.trigger(this.get());
  54. },
  55. onUpdate(updatedOrg: Organization, {replace = false}: UpdateOptions = {}) {
  56. this.loading = false;
  57. this.error = null;
  58. this.errorType = null;
  59. this.organization = replace ? updatedOrg : {...this.organization, ...updatedOrg};
  60. this.dirty = false;
  61. this.trigger(this.get());
  62. },
  63. onFetchOrgError(err: RequestError) {
  64. this.organization = null;
  65. this.errorType = null;
  66. switch (err?.status) {
  67. case 401:
  68. this.errorType = ORGANIZATION_FETCH_ERROR_TYPES.ORG_NO_ACCESS;
  69. break;
  70. case 404:
  71. this.errorType = ORGANIZATION_FETCH_ERROR_TYPES.ORG_NOT_FOUND;
  72. break;
  73. default:
  74. }
  75. this.loading = false;
  76. this.error = err;
  77. this.dirty = false;
  78. this.trigger(this.get());
  79. },
  80. onProjectOrTeamChange() {
  81. // mark the store as dirty so the next fetch will trigger an org details refetch
  82. this.dirty = true;
  83. },
  84. onLoadProjects(projects: Project[]) {
  85. if (this.organization) {
  86. // sort projects to mimic how they are received from backend
  87. projects.sort((a, b) => a.slug.localeCompare(b.slug));
  88. this.organization = {...this.organization, projects};
  89. this.trigger(this.get());
  90. }
  91. },
  92. onLoadTeams(teams: Team[]) {
  93. if (this.organization) {
  94. // sort teams to mimic how they are received from backend
  95. teams.sort((a, b) => a.slug.localeCompare(b.slug));
  96. this.organization = {...this.organization, teams};
  97. this.trigger(this.get());
  98. }
  99. },
  100. get() {
  101. return {
  102. organization: this.organization,
  103. error: this.error,
  104. loading: this.loading,
  105. errorType: this.errorType,
  106. dirty: this.dirty,
  107. };
  108. },
  109. };
  110. type OrganizationStore = Reflux.Store & OrganizationStoreInterface;
  111. const OrganizationStore = Reflux.createStore(storeConfig) as OrganizationStore;
  112. export default OrganizationStore;