reducer.ts 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  1. import { v4 } from 'uuid';
  2. import { Topic, TopicMessage } from 'generated-sources';
  3. import { Action, TopicsState } from 'redux/interfaces';
  4. export const initialState: TopicsState = {
  5. byName: {},
  6. allNames: [],
  7. messages: [],
  8. };
  9. const updateTopicList = (state: TopicsState, payload: Topic[]): TopicsState => {
  10. const initialMemo: TopicsState = {
  11. ...state,
  12. allNames: [],
  13. };
  14. return payload.reduce(
  15. (memo: TopicsState, topic) => ({
  16. ...memo,
  17. byName: {
  18. ...memo.byName,
  19. [topic.name]: {
  20. ...memo.byName[topic.name],
  21. ...topic,
  22. id: v4(),
  23. },
  24. },
  25. allNames: [...memo.allNames, topic.name],
  26. }),
  27. initialMemo
  28. );
  29. };
  30. const addToTopicList = (state: TopicsState, payload: Topic): TopicsState => {
  31. const newState: TopicsState = {
  32. ...state,
  33. };
  34. newState.allNames.push(payload.name);
  35. newState.byName[payload.name] = { ...payload, id: v4() };
  36. return newState;
  37. };
  38. const transformTopicMessages = (
  39. state: TopicsState,
  40. messages: TopicMessage[]
  41. ): TopicsState => ({
  42. ...state,
  43. messages: messages.map((mes) => {
  44. const { content } = mes;
  45. let parsedContent = content;
  46. if (content) {
  47. try {
  48. parsedContent =
  49. typeof content !== 'object' ? JSON.parse(content) : content;
  50. } catch (_) {
  51. // do nothing
  52. }
  53. }
  54. return {
  55. ...mes,
  56. content: parsedContent,
  57. };
  58. }),
  59. });
  60. const reducer = (state = initialState, action: Action): TopicsState => {
  61. switch (action.type) {
  62. case 'GET_TOPICS__SUCCESS':
  63. return updateTopicList(state, action.payload);
  64. case 'GET_TOPIC_DETAILS__SUCCESS':
  65. return {
  66. ...state,
  67. byName: {
  68. ...state.byName,
  69. [action.payload.topicName]: {
  70. ...state.byName[action.payload.topicName],
  71. ...action.payload.details,
  72. },
  73. },
  74. };
  75. case 'GET_TOPIC_MESSAGES__SUCCESS':
  76. return transformTopicMessages(state, action.payload);
  77. case 'GET_TOPIC_CONFIG__SUCCESS':
  78. return {
  79. ...state,
  80. byName: {
  81. ...state.byName,
  82. [action.payload.topicName]: {
  83. ...state.byName[action.payload.topicName],
  84. config: action.payload.config.map((inputConfig) => ({
  85. ...inputConfig,
  86. id: v4(),
  87. })),
  88. },
  89. },
  90. };
  91. case 'POST_TOPIC__SUCCESS':
  92. return addToTopicList(state, action.payload);
  93. default:
  94. return state;
  95. }
  96. };
  97. export default reducer;