2017-09-22 17:52:06 +00:00
|
|
|
/*
|
|
|
|
Copyright 2017 New Vector Ltd
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import EventEmitter from 'events';
|
2017-10-23 15:04:26 +00:00
|
|
|
import { groupMemberFromApiObject, groupRoomFromApiObject } from '../groups';
|
2017-10-23 14:28:38 +00:00
|
|
|
import FlairStore from './FlairStore';
|
2017-11-28 11:54:05 +00:00
|
|
|
import MatrixClientPeg from '../MatrixClientPeg';
|
2017-09-22 17:52:06 +00:00
|
|
|
|
2018-01-17 16:59:13 +00:00
|
|
|
function parseMembersResponse(response) {
|
|
|
|
return response.chunk.map((apiMember) => groupMemberFromApiObject(apiMember));
|
|
|
|
}
|
|
|
|
|
|
|
|
function parseRoomsResponse(response) {
|
|
|
|
return response.chunk.map((apiRoom) => groupRoomFromApiObject(apiRoom));
|
|
|
|
}
|
|
|
|
|
2018-03-13 11:35:15 +00:00
|
|
|
// The number of ongoing group requests
|
|
|
|
let ongoingRequestCount = 0;
|
|
|
|
|
|
|
|
// This has arbitrarily been set to a small number to lower the priority
|
|
|
|
// of doing group-related requests because we care about other important
|
|
|
|
// requests like hitting /sync.
|
|
|
|
const LIMIT = 3; // Maximum number of ongoing group requests
|
|
|
|
|
|
|
|
// FIFO queue of functions to call in the backlog
|
|
|
|
const backlogQueue = [
|
|
|
|
// () => {...}
|
|
|
|
];
|
|
|
|
|
|
|
|
// Pull from the FIFO queue
|
|
|
|
function checkBacklog() {
|
|
|
|
const item = backlogQueue.shift();
|
|
|
|
if (typeof item === 'function') item();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Limit the maximum number of ongoing promises returned by fn to LIMIT and
|
|
|
|
// use a FIFO queue to handle the backlog.
|
|
|
|
function limitConcurrency(fn) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const item = () => {
|
|
|
|
ongoingRequestCount++;
|
|
|
|
resolve();
|
|
|
|
};
|
|
|
|
if (ongoingRequestCount >= LIMIT) {
|
|
|
|
// Enqueue this request for later execution
|
|
|
|
backlogQueue.push(item);
|
|
|
|
} else {
|
|
|
|
item();
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.then(fn)
|
2018-05-01 13:04:13 +00:00
|
|
|
.catch((err) => {
|
|
|
|
ongoingRequestCount--;
|
|
|
|
checkBacklog();
|
|
|
|
throw err;
|
|
|
|
})
|
2018-03-13 11:35:15 +00:00
|
|
|
.then((result) => {
|
|
|
|
ongoingRequestCount--;
|
|
|
|
checkBacklog();
|
|
|
|
return result;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-09-22 17:52:06 +00:00
|
|
|
/**
|
2018-05-01 10:18:45 +00:00
|
|
|
* Global store for tracking group summary, members, invited members and rooms.
|
2017-09-22 17:52:06 +00:00
|
|
|
*/
|
2018-05-01 10:18:45 +00:00
|
|
|
class GroupStore extends EventEmitter {
|
|
|
|
STATE_KEY = {
|
2017-10-31 16:13:13 +00:00
|
|
|
GroupMembers: 'GroupMembers',
|
|
|
|
GroupInvitedMembers: 'GroupInvitedMembers',
|
|
|
|
Summary: 'Summary',
|
|
|
|
GroupRooms: 'GroupRooms',
|
|
|
|
};
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
constructor() {
|
2017-09-22 17:52:06 +00:00
|
|
|
super();
|
2018-01-17 16:59:13 +00:00
|
|
|
this._state = {};
|
2018-05-01 10:18:45 +00:00
|
|
|
this._state[this.STATE_KEY.Summary] = {};
|
|
|
|
this._state[this.STATE_KEY.GroupRooms] = {};
|
|
|
|
this._state[this.STATE_KEY.GroupMembers] = {};
|
|
|
|
this._state[this.STATE_KEY.GroupInvitedMembers] = {};
|
|
|
|
|
2017-10-31 11:42:09 +00:00
|
|
|
this._ready = {};
|
2018-05-01 10:18:45 +00:00
|
|
|
this._ready[this.STATE_KEY.Summary] = {};
|
|
|
|
this._ready[this.STATE_KEY.GroupRooms] = {};
|
|
|
|
this._ready[this.STATE_KEY.GroupMembers] = {};
|
|
|
|
this._ready[this.STATE_KEY.GroupInvitedMembers] = {};
|
|
|
|
|
|
|
|
this._fetchResourcePromise = {
|
|
|
|
[this.STATE_KEY.Summary]: {},
|
|
|
|
[this.STATE_KEY.GroupRooms]: {},
|
|
|
|
[this.STATE_KEY.GroupMembers]: {},
|
|
|
|
[this.STATE_KEY.GroupInvitedMembers]: {},
|
|
|
|
};
|
2017-10-27 10:36:32 +00:00
|
|
|
|
2018-01-17 16:59:13 +00:00
|
|
|
this._resourceFetcher = {
|
2018-05-01 10:18:45 +00:00
|
|
|
[this.STATE_KEY.Summary]: (groupId) => {
|
2018-03-13 11:35:15 +00:00
|
|
|
return limitConcurrency(
|
2018-05-01 10:18:45 +00:00
|
|
|
() => MatrixClientPeg.get().getGroupSummary(groupId),
|
2018-03-13 11:35:15 +00:00
|
|
|
);
|
2018-01-17 16:59:13 +00:00
|
|
|
},
|
2018-05-01 10:18:45 +00:00
|
|
|
[this.STATE_KEY.GroupRooms]: (groupId) => {
|
2018-03-13 11:35:15 +00:00
|
|
|
return limitConcurrency(
|
2018-05-01 10:18:45 +00:00
|
|
|
() => MatrixClientPeg.get().getGroupRooms(groupId).then(parseRoomsResponse),
|
2018-03-13 11:35:15 +00:00
|
|
|
);
|
2018-01-17 16:59:13 +00:00
|
|
|
},
|
2018-05-01 10:18:45 +00:00
|
|
|
[this.STATE_KEY.GroupMembers]: (groupId) => {
|
2018-03-13 11:35:15 +00:00
|
|
|
return limitConcurrency(
|
2018-05-01 10:18:45 +00:00
|
|
|
() => MatrixClientPeg.get().getGroupUsers(groupId).then(parseMembersResponse),
|
2018-03-13 11:35:15 +00:00
|
|
|
);
|
2018-01-17 16:59:13 +00:00
|
|
|
},
|
2018-05-01 10:18:45 +00:00
|
|
|
[this.STATE_KEY.GroupInvitedMembers]: (groupId) => {
|
2018-03-13 11:35:15 +00:00
|
|
|
return limitConcurrency(
|
2018-05-01 10:18:45 +00:00
|
|
|
() => MatrixClientPeg.get().getGroupInvitedUsers(groupId).then(parseMembersResponse),
|
2018-03-13 11:35:15 +00:00
|
|
|
);
|
2018-01-17 16:59:13 +00:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
this.on('error', (err, groupId) => {
|
|
|
|
console.error(`GroupStore encountered error whilst fetching data for ${groupId}`, err);
|
2017-10-27 10:36:32 +00:00
|
|
|
});
|
2017-10-23 15:04:26 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
_fetchResource(stateKey, groupId) {
|
2018-01-17 17:01:42 +00:00
|
|
|
// Ongoing request, ignore
|
2018-05-01 10:18:45 +00:00
|
|
|
if (this._fetchResourcePromise[stateKey][groupId]) return;
|
2018-01-17 17:01:42 +00:00
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
const clientPromise = this._resourceFetcher[stateKey](groupId);
|
2018-01-17 17:01:42 +00:00
|
|
|
|
|
|
|
// Indicate ongoing request
|
2018-05-01 10:18:45 +00:00
|
|
|
this._fetchResourcePromise[stateKey][groupId] = clientPromise;
|
2018-01-17 17:01:42 +00:00
|
|
|
|
2018-01-17 16:59:13 +00:00
|
|
|
clientPromise.then((result) => {
|
2018-05-01 10:18:45 +00:00
|
|
|
this._state[stateKey][groupId] = result;
|
|
|
|
console.info(this._state);
|
|
|
|
this._ready[stateKey][groupId] = true;
|
2017-10-23 15:04:26 +00:00
|
|
|
this._notifyListeners();
|
|
|
|
}).catch((err) => {
|
2017-10-27 10:37:45 +00:00
|
|
|
// Invited users not visible to non-members
|
2018-05-01 10:18:45 +00:00
|
|
|
if (stateKey === this.STATE_KEY.GroupInvitedMembers && err.httpStatus === 403) {
|
2017-10-27 10:37:45 +00:00
|
|
|
return;
|
|
|
|
}
|
2017-09-22 17:52:06 +00:00
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
console.error(`Failed to get resource ${stateKey} for ${groupId}`, err);
|
|
|
|
this.emit('error', err, groupId);
|
2018-01-17 17:01:42 +00:00
|
|
|
}).finally(() => {
|
|
|
|
// Indicate finished request, allow for future fetches
|
2018-05-01 10:18:45 +00:00
|
|
|
delete this._fetchResourcePromise[stateKey][groupId];
|
2017-09-22 17:52:06 +00:00
|
|
|
});
|
|
|
|
|
2018-01-17 16:59:13 +00:00
|
|
|
return clientPromise;
|
2017-10-05 13:30:04 +00:00
|
|
|
}
|
|
|
|
|
2017-09-22 17:52:06 +00:00
|
|
|
_notifyListeners() {
|
|
|
|
this.emit('update');
|
|
|
|
}
|
|
|
|
|
2018-01-02 18:12:08 +00:00
|
|
|
/**
|
|
|
|
* Register a listener to recieve updates from the store. This also
|
|
|
|
* immediately triggers an update to send the current state of the
|
|
|
|
* store (which could be the initial state).
|
|
|
|
*
|
2018-05-01 10:18:45 +00:00
|
|
|
* This also causes a fetch of all data of the specified group,
|
|
|
|
* which might cause 4 separate HTTP requests, but only if said
|
|
|
|
* requests aren't already ongoing.
|
2018-01-02 18:12:08 +00:00
|
|
|
*
|
2018-05-01 10:18:45 +00:00
|
|
|
* @param {string} groupId the ID of the group to fetch data for.
|
2018-01-02 18:12:08 +00:00
|
|
|
* @param {function} fn the function to call when the store updates.
|
|
|
|
* @return {Object} tok a registration "token" with a single
|
|
|
|
* property `unregister`, a function that can
|
|
|
|
* be called to unregister the listener such
|
|
|
|
* that it won't be called any more.
|
|
|
|
*/
|
2018-05-01 10:18:45 +00:00
|
|
|
registerListener(groupId, fn) {
|
2017-10-27 10:36:32 +00:00
|
|
|
this.on('update', fn);
|
2017-10-31 11:42:09 +00:00
|
|
|
// Call to set initial state (before fetching starts)
|
|
|
|
this.emit('update');
|
2018-01-17 16:59:13 +00:00
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
this._fetchResource(this.STATE_KEY.Summary, groupId);
|
|
|
|
this._fetchResource(this.STATE_KEY.GroupRooms, groupId);
|
|
|
|
this._fetchResource(this.STATE_KEY.GroupMembers, groupId);
|
|
|
|
this._fetchResource(this.STATE_KEY.GroupInvitedMembers, groupId);
|
2017-12-15 14:12:21 +00:00
|
|
|
|
2018-01-02 18:12:08 +00:00
|
|
|
// Similar to the Store of flux/utils, we return a "token" that
|
|
|
|
// can be used to unregister the listener.
|
2017-12-15 14:12:21 +00:00
|
|
|
return {
|
|
|
|
unregister: () => {
|
|
|
|
this.unregisterListener(fn);
|
|
|
|
},
|
|
|
|
};
|
2017-10-27 10:36:32 +00:00
|
|
|
}
|
|
|
|
|
2017-10-27 13:33:10 +00:00
|
|
|
unregisterListener(fn) {
|
|
|
|
this.removeListener('update', fn);
|
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
isStateReady(groupId, id) {
|
|
|
|
return this._ready[id][groupId];
|
2017-10-31 11:42:09 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
getSummary(groupId) {
|
|
|
|
return this._state[this.STATE_KEY.Summary][groupId] || {};
|
2017-09-22 17:52:06 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
getGroupRooms(groupId) {
|
|
|
|
return this._state[this.STATE_KEY.GroupRooms][groupId] || [];
|
2017-10-05 13:30:04 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
getGroupMembers(groupId) {
|
|
|
|
return this._state[this.STATE_KEY.GroupMembers][groupId] || [];
|
2017-10-23 15:04:26 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
getGroupInvitedMembers(groupId) {
|
|
|
|
return this._state[this.STATE_KEY.GroupInvitedMembers][groupId] || [];
|
2017-10-23 15:04:26 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
getGroupPublicity(groupId) {
|
|
|
|
return (this._state[this.STATE_KEY.Summary][groupId] || {}).user ?
|
|
|
|
(this._state[this.STATE_KEY.Summary][groupId] || {}).user.is_publicised : null;
|
2017-10-17 15:08:19 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
isUserPrivileged(groupId) {
|
|
|
|
return (this._state[this.STATE_KEY.Summary][groupId] || {}).user ?
|
|
|
|
(this._state[this.STATE_KEY.Summary][groupId] || {}).user.is_privileged : null;
|
2017-10-17 15:08:19 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
addRoomToGroup(groupId, roomId, isPublic) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2018-05-01 10:18:45 +00:00
|
|
|
.addRoomToGroup(groupId, roomId, isPublic)
|
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.GroupRooms, groupId));
|
2017-11-02 13:25:55 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
updateGroupRoomVisibility(groupId, roomId, isPublic) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2018-05-01 10:18:45 +00:00
|
|
|
.updateGroupRoomVisibility(groupId, roomId, isPublic)
|
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.GroupRooms, groupId));
|
2017-10-05 13:30:04 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
removeRoomFromGroup(groupId, roomId) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2018-05-01 10:18:45 +00:00
|
|
|
.removeRoomFromGroup(groupId, roomId)
|
2017-10-05 13:30:04 +00:00
|
|
|
// Room might be in the summary, refresh just in case
|
2018-05-01 10:18:45 +00:00
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.Summary, groupId))
|
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.GroupRooms, groupId));
|
2017-10-04 15:56:35 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
inviteUserToGroup(groupId, userId) {
|
|
|
|
return MatrixClientPeg.get().inviteUserToGroup(groupId, userId)
|
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.GroupInvitedMembers, groupId));
|
2017-10-23 15:04:26 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
acceptGroupInvite(groupId) {
|
|
|
|
return MatrixClientPeg.get().acceptGroupInvite(groupId)
|
2018-04-10 09:03:54 +00:00
|
|
|
// The user should now be able to access (personal) group settings
|
2018-05-01 10:18:45 +00:00
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.Summary, groupId))
|
2018-04-10 09:03:54 +00:00
|
|
|
// The user might be able to see more rooms now
|
2018-05-01 10:18:45 +00:00
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.GroupRooms, groupId))
|
2018-04-10 09:03:54 +00:00
|
|
|
// The user should now appear as a member
|
2018-05-01 10:18:45 +00:00
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.GroupMembers, groupId))
|
2018-04-10 09:03:54 +00:00
|
|
|
// The user should now not appear as an invited member
|
2018-05-01 10:18:45 +00:00
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.GroupInvitedMembers, groupId));
|
2018-04-10 09:03:54 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
joinGroup(groupId) {
|
|
|
|
return MatrixClientPeg.get().joinGroup(groupId)
|
2018-04-10 09:03:54 +00:00
|
|
|
// The user should now be able to access (personal) group settings
|
2018-05-01 10:18:45 +00:00
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.Summary, groupId))
|
2017-11-07 18:51:41 +00:00
|
|
|
// The user might be able to see more rooms now
|
2018-05-01 10:18:45 +00:00
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.GroupRooms, groupId))
|
2017-11-08 11:52:52 +00:00
|
|
|
// The user should now appear as a member
|
2018-05-01 10:18:45 +00:00
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.GroupMembers, groupId))
|
2018-01-17 16:59:13 +00:00
|
|
|
// The user should now not appear as an invited member
|
2018-05-01 10:18:45 +00:00
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.GroupInvitedMembers, groupId));
|
2017-11-07 18:51:41 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
leaveGroup(groupId) {
|
|
|
|
return MatrixClientPeg.get().leaveGroup(groupId)
|
2018-04-10 09:03:54 +00:00
|
|
|
// The user should now not be able to access group settings
|
2018-05-01 10:18:45 +00:00
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.Summary, groupId))
|
2018-04-10 09:03:54 +00:00
|
|
|
// The user might only be able to see a subset of rooms now
|
2018-05-01 10:18:45 +00:00
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.GroupRooms, groupId))
|
2018-04-10 09:03:54 +00:00
|
|
|
// The user should now not appear as a member
|
2018-05-01 10:18:45 +00:00
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.GroupMembers, groupId));
|
2018-04-10 09:03:54 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
addRoomToGroupSummary(groupId, roomId, categoryId) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2018-05-01 10:18:45 +00:00
|
|
|
.addRoomToGroupSummary(groupId, roomId, categoryId)
|
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.Summary, groupId));
|
2017-09-22 17:52:06 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
addUserToGroupSummary(groupId, userId, roleId) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2018-05-01 10:18:45 +00:00
|
|
|
.addUserToGroupSummary(groupId, userId, roleId)
|
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.Summary, groupId));
|
2017-09-22 17:52:06 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
removeRoomFromGroupSummary(groupId, roomId) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2018-05-01 10:18:45 +00:00
|
|
|
.removeRoomFromGroupSummary(groupId, roomId)
|
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.Summary, groupId));
|
2017-09-22 17:52:06 +00:00
|
|
|
}
|
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
removeUserFromGroupSummary(groupId, userId) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2018-05-01 10:18:45 +00:00
|
|
|
.removeUserFromGroupSummary(groupId, userId)
|
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.Summary, groupId));
|
2017-09-22 17:52:06 +00:00
|
|
|
}
|
2017-09-26 13:46:57 +00:00
|
|
|
|
2018-05-01 10:18:45 +00:00
|
|
|
setGroupPublicity(groupId, isPublished) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2018-05-01 10:18:45 +00:00
|
|
|
.setGroupPublicity(groupId, isPublished)
|
2017-11-28 11:54:05 +00:00
|
|
|
.then(() => { FlairStore.invalidatePublicisedGroups(MatrixClientPeg.get().credentials.userId); })
|
2018-05-01 10:18:45 +00:00
|
|
|
.then(this._fetchResource.bind(this, this.STATE_KEY.Summary, groupId));
|
2017-09-26 13:46:57 +00:00
|
|
|
}
|
2017-09-22 17:52:06 +00:00
|
|
|
}
|
2018-05-01 10:18:45 +00:00
|
|
|
|
|
|
|
let singletonGroupStore = null;
|
|
|
|
if (!singletonGroupStore) {
|
|
|
|
singletonGroupStore = new GroupStore();
|
|
|
|
}
|
|
|
|
module.exports = singletonGroupStore;
|