Collection model

This commit is contained in:
Jori Lallo
2017-05-23 00:45:15 -07:00
parent 07ff5b6ae5
commit 6b566a568f
6 changed files with 116 additions and 16 deletions

View File

@ -0,0 +1,50 @@
// @flow
import { extendObservable, action, computed, runInAction } from 'mobx';
import invariant from 'invariant';
import _ from 'lodash';
import ApiClient, { client } from 'utils/ApiClient';
import stores from 'stores';
import ErrorsStore from 'stores/ErrorsStore';
import type { NavigationNode } from 'types';
class Collection {
createdAt: string;
description: ?string;
id: string;
name: string;
type: 'atlas' | 'journal';
navigationTree: NavigationNode;
updatedAt: string;
url: string;
client: ApiClient;
errors: ErrorsStore;
/* Actions */
@action update = async () => {
try {
const res = await this.client.post('/collections.info', { id: this.id });
invariant(res && res.data, 'API response should be available');
const { data } = res;
runInAction('Collection#update', () => {
this.updateData(data);
});
} catch (e) {
this.errors.add('Collection failed loading');
}
};
updateData(data: Collection) {
extendObservable(this, data);
}
constructor(collection: Collection) {
this.updateData(collection);
this.client = client;
this.errors = stores.errors;
}
}
export default Collection;

View File

@ -0,0 +1,58 @@
/* eslint-disable */
import Collection from './Collection';
jest.mock('utils/ApiClient', () => ({
client: { post: {} },
}));
jest.mock('stores', () => ({ errors: {} }));
describe('Collection model', () => {
test('should initialize with data', () => {
const collection = new Collection({
id: 123,
name: 'Engineering',
});
expect(collection.name).toBe('Engineering');
});
describe('#update', () => {
test('should update', async () => {
const collection = new Collection({
id: 123,
name: 'Engineering',
});
collection.client = {
post: jest.fn(() => ({
data: {
name: 'New collection',
},
})),
};
await collection.update();
expect(collection.client.post).toHaveBeenCalledWith('/collections.info', {
id: 123,
});
expect(collection.name).toBe('New collection');
});
test('should report errors', async () => {
const collection = new Collection({
id: 123,
});
collection.client = {
post: jest.fn(() => Promise.reject),
};
collection.errors = {
add: jest.fn(),
};
await collection.update();
expect(collection.errors.add).toHaveBeenCalledWith(
'Collection failed loading'
);
});
});
});

View File

@ -2,7 +2,7 @@
import { observable, action, computed } from 'mobx'; import { observable, action, computed } from 'mobx';
import invariant from 'invariant'; import invariant from 'invariant';
import { client } from 'utils/ApiClient'; import { client } from 'utils/ApiClient';
import type { Collection } from 'types'; import Collection from 'models/Collection';
const store = new class AtlasStore { const store = new class AtlasStore {
@observable collection: ?(Collection & { recentDocuments?: Object[] }); @observable collection: ?(Collection & { recentDocuments?: Object[] });
@ -25,7 +25,7 @@ const store = new class AtlasStore {
const res = await client.get('/collections.info', { id }); const res = await client.get('/collections.info', { id });
invariant(res && res.data, 'Data should be available'); invariant(res && res.data, 'Data should be available');
const { data } = res; const { data } = res;
this.collection = data; this.collection = new Collection(data);
successCallback(data); successCallback(data);
} catch (e) { } catch (e) {
console.error('Something went wrong'); console.error('Something went wrong');

View File

@ -2,7 +2,8 @@
import { observable, action, runInAction } from 'mobx'; import { observable, action, runInAction } from 'mobx';
import invariant from 'invariant'; import invariant from 'invariant';
import { client } from 'utils/ApiClient'; import { client } from 'utils/ApiClient';
import type { Pagination, Collection } from 'types'; import type { Pagination } from 'types';
import Collection from 'models/Collection';
type Options = { type Options = {
team: Object, team: Object,
@ -30,7 +31,7 @@ class DashboardStore {
); );
const { data, pagination } = res; const { data, pagination } = res;
runInAction('fetchCollections', () => { runInAction('fetchCollections', () => {
this.collections = data; this.collections = data.map(collection => new Collection(collection));
this.pagination = pagination; this.pagination = pagination;
}); });
} catch (e) { } catch (e) {

View File

@ -2,10 +2,12 @@
import { autorunAsync } from 'mobx'; import { autorunAsync } from 'mobx';
import UserStore, { USER_STORE } from './UserStore'; import UserStore, { USER_STORE } from './UserStore';
import UiStore, { UI_STORE } from './UiStore'; import UiStore, { UI_STORE } from './UiStore';
import ErrorsStore from './ErrorsStore';
const stores = { const stores = {
user: new UserStore(), user: new UserStore(),
ui: new UiStore(), ui: new UiStore(),
errors: new ErrorsStore(),
}; };
// Persist stores to localStorage // Persist stores to localStorage

View File

@ -19,20 +19,9 @@ export type NavigationNode = {
children: Array<NavigationNode>, children: Array<NavigationNode>,
}; };
export type Collection = {
createdAt: string,
description: ?string,
id: string,
name: string,
type: 'atlas' | 'journal',
navigationTree: NavigationNode,
updatedAt: string,
url: string,
};
export type Document = { export type Document = {
collaborators: Array<User>, collaborators: Array<User>,
collection: Collection, collection: Object,
createdAt: string, createdAt: string,
createdBy: string, createdBy: string,
html: string, html: string,