import { Map as ImmutableMap, OrderedSet as ImmutableOrderedSet, fromJS, is, } from 'immutable'; import { STATUS_IMPORT } from 'soapbox/actions/importer'; import { CONTEXT_FETCH_SUCCESS } from 'soapbox/actions/statuses'; import { TIMELINE_DELETE } from 'soapbox/actions/timelines'; import { applyActions } from 'soapbox/jest/test-helpers'; import reducer, { ReducerRecord } from '../contexts'; describe('contexts reducer', () => { it('should return the initial state', () => { expect(reducer(undefined, {} as any)).toEqual(ReducerRecord({ inReplyTos: ImmutableMap(), replies: ImmutableMap(), })); }); describe(CONTEXT_FETCH_SUCCESS, () => { it('inserts a tombstone connecting an orphaned descendant', () => { const status = { id: 'A', in_reply_to_id: null }; const context = { id: 'A', ancestors: [], descendants: [ { id: 'C', in_reply_to_id: 'B' }, ], }; const actions = [ { type: STATUS_IMPORT, status }, { type: CONTEXT_FETCH_SUCCESS, ...context }, ]; const result = applyActions(undefined, actions, reducer); expect(result.inReplyTos.get('C')).toBe('C-tombstone'); expect(result.replies.get('A').toArray()).toEqual(['C-tombstone']); }); it('inserts a tombstone connecting an orphaned descendant (with null in_reply_to_id)', () => { const status = { id: 'A', in_reply_to_id: null }; const context = { id: 'A', ancestors: [], descendants: [ { id: 'C', in_reply_to_id: null }, ], }; const actions = [ { type: STATUS_IMPORT, status }, { type: CONTEXT_FETCH_SUCCESS, ...context }, ]; const result = applyActions(undefined, actions, reducer); expect(result.inReplyTos.get('C')).toBe('C-tombstone'); expect(result.replies.get('A').toArray()).toEqual(['C-tombstone']); }); it('doesn\'t explode when it encounters a loop', () => { const status = { id: 'A', in_reply_to_id: null }; const context = { id: 'A', ancestors: [], descendants: [ { id: 'C', in_reply_to_id: 'E' }, { id: 'D', in_reply_to_id: 'C' }, { id: 'E', in_reply_to_id: 'D' }, { id: 'F', in_reply_to_id: 'F' }, ], }; const actions = [ { type: STATUS_IMPORT, status }, { type: CONTEXT_FETCH_SUCCESS, ...context }, ]; const result = applyActions(undefined, actions, reducer); // These checks are superficial. We just don't want a stack overflow! expect(result.inReplyTos.get('C')).toBe('C-tombstone'); expect(result.replies.get('A').toArray()).toEqual(['C-tombstone', 'F-tombstone']); }); }); describe(TIMELINE_DELETE, () => { it('deletes the status', () => { const action = { type: TIMELINE_DELETE, id: 'B' }; const state = ReducerRecord({ inReplyTos: fromJS({ B: 'A', C: 'B', }) as ImmutableMap<string, string>, replies: fromJS({ A: ImmutableOrderedSet(['B']), B: ImmutableOrderedSet(['C']), }) as ImmutableMap<string, ImmutableOrderedSet<string>>, }); const expected = ReducerRecord({ inReplyTos: fromJS({}) as ImmutableMap<string, string>, replies: fromJS({ A: ImmutableOrderedSet(), }) as ImmutableMap<string, ImmutableOrderedSet<string>>, }); expect(is(reducer(state, action), expected)).toBe(true); }); }); });