Collection model
This commit is contained in:
50
frontend/models/Collection.js
Normal file
50
frontend/models/Collection.js
Normal 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;
|
58
frontend/models/Collection.test.js
Normal file
58
frontend/models/Collection.test.js
Normal 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'
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
@ -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');
|
||||||
|
@ -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) {
|
||||||
|
@ -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
|
||||||
|
@ -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,
|
||||||
|
Reference in New Issue
Block a user