import { AsyncStorage, InteractionManager } from 'react-native'; import semver from 'semver'; import { Rocketchat as RocketchatClient, settings as RocketChatSettings } from '@rocket.chat/sdk'; import RNUserDefaults from 'rn-user-defaults'; import { Q } from '@nozbe/watermelondb'; import * as FileSystem from 'expo-file-system'; import reduxStore from './createStore'; import defaultSettings from '../constants/settings'; import messagesStatus from '../constants/messagesStatus'; import database from './database'; import log from '../utils/log'; import { isIOS, getBundleId } from '../utils/deviceInfo'; import { extractHostname } from '../utils/server'; import fetch, { headers } from '../utils/fetch'; import { setUser, setLoginServices, loginRequest } from '../actions/login'; import { disconnect, connectSuccess, connectRequest } from '../actions/connect'; import { shareSelectServer, shareSetUser } from '../actions/share'; import subscribeRooms from './methods/subscriptions/rooms'; import subscribeRoom from './methods/subscriptions/room'; import protectedFunction from './methods/helpers/protectedFunction'; import readMessages from './methods/readMessages'; import getSettings, { setSettings } from './methods/getSettings'; import getRooms from './methods/getRooms'; import getPermissions from './methods/getPermissions'; import { getCustomEmojis, setCustomEmojis } from './methods/getCustomEmojis'; import getSlashCommands from './methods/getSlashCommands'; import getRoles from './methods/getRoles'; import canOpenRoom from './methods/canOpenRoom'; import loadMessagesForRoom from './methods/loadMessagesForRoom'; import loadMissedMessages from './methods/loadMissedMessages'; import loadThreadMessages from './methods/loadThreadMessages'; import sendMessage, { sendMessageCall } from './methods/sendMessage'; import { sendFileMessage, cancelUpload, isUploadActive } from './methods/sendFileMessage'; import callJitsi from './methods/callJitsi'; import { getDeviceToken } from '../notifications/push'; import { SERVERS, SERVER_URL } from '../constants/userDefaults'; import { setActiveUsers } from '../actions/activeUsers'; import I18n from '../i18n'; const TOKEN_KEY = 'reactnativemeteor_usertoken'; const SORT_PREFS_KEY = 'RC_SORT_PREFS_KEY'; export const MARKDOWN_KEY = 'RC_MARKDOWN_KEY'; export const THEME_PREFERENCES_KEY = 'RC_THEME_PREFERENCES_KEY'; export const CRASH_REPORT_KEY = 'RC_CRASH_REPORT_KEY'; const returnAnArray = obj => obj || []; const MIN_ROCKETCHAT_VERSION = '0.70.0'; const STATUSES = ['offline', 'online', 'away', 'busy']; RocketChatSettings.customHeaders = headers; const RocketChat = { TOKEN_KEY, callJitsi, async subscribeRooms() { if (this.roomsSub) { this.roomsSub.stop(); } try { this.roomsSub = await subscribeRooms.call(this); } catch (e) { log(e); } }, subscribeRoom, canOpenRoom, createChannel({ name, users, type, readOnly, broadcast }) { // RC 0.51.0 return this.sdk.methodCall(type ? 'createPrivateGroup' : 'createChannel', name, users, readOnly, {}, { broadcast }); }, async getUserToken() { try { return await RNUserDefaults.get(TOKEN_KEY); } catch (error) { console.warn(`RNUserDefaults error: ${ error.message }`); } }, async getWebsocketInfo({ server }) { // Use useSsl: false only if server url starts with http:// const useSsl = !/http:\/\//.test(server); const sdk = new RocketchatClient({ host: server, protocol: 'ddp', useSsl }); try { await sdk.connect(); } catch (err) { if (err.message && err.message.includes('400')) { return { success: false, message: 'Websocket_disabled', messageOptions: { contact: I18n.t('Contact_your_server_admin') } }; } } sdk.disconnect(); return { success: true }; }, async getServerInfo(server) { const notRCServer = { success: false, message: 'Not_RC_Server', messageOptions: { contact: I18n.t('Contact_your_server_admin') } }; try { const result = await fetch(`${ server }/api/info`).then(async(response) => { let res = notRCServer; try { res = await response.json(); if (!(res && res.success)) { return notRCServer; } } catch (e) { // do nothing } return res; }); if (result.success) { if (semver.lt(result.version, MIN_ROCKETCHAT_VERSION)) { return { success: false, message: 'Invalid_server_version', messageOptions: { currentVersion: result.version, minVersion: MIN_ROCKETCHAT_VERSION } }; } } return result; } catch (e) { log(e); } return { success: false, message: 'The_URL_is_invalid', messageOptions: { contact: I18n.t('Contact_your_server_admin') } }; }, stopListener(listener) { return listener && listener.stop(); }, connect({ server, user, logoutOnError = false }) { return new Promise((resolve) => { if (!this.sdk || this.sdk.client.host !== server) { database.setActiveDB(server); } reduxStore.dispatch(connectRequest()); if (this.connectTimeout) { clearTimeout(this.connectTimeout); } if (this.connectedListener) { this.connectedListener.then(this.stopListener); } if (this.closeListener) { this.closeListener.then(this.stopListener); } if (this.usersListener) { this.usersListener.then(this.stopListener); } if (this.notifyLoggedListener) { this.notifyLoggedListener.then(this.stopListener); } if (this.roomsSub) { this.roomsSub.stop(); } if (this.sdk) { this.sdk.disconnect(); this.sdk = null; } // Use useSsl: false only if server url starts with http:// const useSsl = !/http:\/\//.test(server); this.sdk = new RocketchatClient({ host: server, protocol: 'ddp', useSsl }); this.getSettings(); this.sdk.connect() .then(() => { if (user && user.token) { reduxStore.dispatch(loginRequest({ resume: user.token }, logoutOnError)); } }) .catch((err) => { console.log('connect error', err); // when `connect` raises an error, we try again in 10 seconds this.connectTimeout = setTimeout(() => { this.connect({ server, user }); }, 10000); }); this.connectedListener = this.sdk.onStreamData('connected', () => { reduxStore.dispatch(connectSuccess()); }); this.closeListener = this.sdk.onStreamData('close', () => { reduxStore.dispatch(disconnect()); }); this.usersListener = this.sdk.onStreamData('users', protectedFunction(ddpMessage => RocketChat._setUser(ddpMessage))); this.notifyLoggedListener = this.sdk.onStreamData('stream-notify-logged', protectedFunction((ddpMessage) => { const { eventName } = ddpMessage.fields; if (eventName === 'user-status') { this.activeUsers = this.activeUsers || {}; if (!this._setUserTimer) { this._setUserTimer = setTimeout(() => { const activeUsersBatch = this.activeUsers; InteractionManager.runAfterInteractions(() => { reduxStore.dispatch(setActiveUsers(activeUsersBatch)); }); this._setUserTimer = null; return this.activeUsers = {}; }, 10000); } const userStatus = ddpMessage.fields.args[0]; const [id,, status] = userStatus; this.activeUsers[id] = STATUSES[status]; const { user: loggedUser } = reduxStore.getState().login; if (loggedUser && loggedUser.id === id) { reduxStore.dispatch(setUser({ status: STATUSES[status] })); } } })); resolve(); }); }, async shareExtensionInit(server) { database.setShareDB(server); if (this.shareSDK) { this.shareSDK.disconnect(); this.shareSDK = null; } // Use useSsl: false only if server url starts with http:// const useSsl = !/http:\/\//.test(server); this.shareSDK = new RocketchatClient({ host: server, protocol: 'ddp', useSsl }); // set Server const serversDB = database.servers; reduxStore.dispatch(shareSelectServer(server)); // set User info try { const userId = await RNUserDefaults.get(`${ RocketChat.TOKEN_KEY }-${ server }`); const userCollections = serversDB.collections.get('users'); let user = null; if (userId) { const userRecord = await userCollections.find(userId); user = { id: userRecord.id, token: userRecord.token, username: userRecord.username }; } reduxStore.dispatch(shareSetUser({ id: user.id, token: user.token, username: user.username })); await RocketChat.login({ resume: user.token }); } catch (e) { log(e); } }, closeShareExtension() { if (this.shareSDK) { this.shareSDK.disconnect(); this.shareSDK = null; } database.share = null; }, updateJitsiTimeout(rid) { return this.sdk.methodCall('jitsi:updateTimeout', rid); }, register(credentials) { // RC 0.50.0 return this.sdk.post('users.register', credentials, false); }, setUsername(username) { // RC 0.51.0 return this.sdk.methodCall('setUsername', username); }, forgotPassword(email) { // RC 0.64.0 return this.sdk.post('users.forgotPassword', { email }, false); }, async loginWithPassword({ user, password, code }) { let params = { user, password }; const state = reduxStore.getState(); if (state.settings.LDAP_Enable) { params = { username: user, ldapPass: password, ldap: true, ldapOptions: {} }; } else if (state.settings.CROWD_Enable) { params = { username: user, crowdPassword: password, crowd: true }; } if (code) { params = { user, password, code }; } try { return await this.login(params); } catch (error) { throw error; } }, async loginOAuthOrSso(params) { try { const result = await this.login(params); reduxStore.dispatch(loginRequest({ resume: result.token })); } catch (error) { throw error; } }, async login(params) { try { const sdk = this.shareSDK || this.sdk; // RC 0.64.0 await sdk.login(params); const { result } = sdk.currentLogin; const user = { id: result.userId, token: result.authToken, username: result.me.username, name: result.me.name, language: result.me.language, status: result.me.status, customFields: result.me.customFields, emails: result.me.emails, roles: result.me.roles }; return user; } catch (e) { throw e; } }, async logout({ server }) { if (this.roomsSub) { this.roomsSub.stop(); } if (this.activeUsersSubTimeout) { clearTimeout(this.activeUsersSubTimeout); this.activeUsersSubTimeout = false; } try { await this.removePushToken(); } catch (error) { console.log('logout -> removePushToken -> catch -> error', error); } try { // RC 0.60.0 await this.sdk.logout(); } catch (error) { console.log('​logout -> api logout -> catch -> error', error); } this.sdk = null; try { const servers = await RNUserDefaults.objectForKey(SERVERS); await RNUserDefaults.setObjectForKey(SERVERS, servers && servers.filter(srv => srv[SERVER_URL] !== server)); // clear certificate for server - SSL Pinning const certificate = await RNUserDefaults.objectForKey(extractHostname(server)); if (certificate && certificate.path) { await RNUserDefaults.clear(extractHostname(server)); await FileSystem.deleteAsync(certificate.path); } } catch (error) { console.log('logout_rn_user_defaults', error); } const userId = await RNUserDefaults.get(`${ TOKEN_KEY }-${ server }`); try { const serversDB = database.servers; await serversDB.action(async() => { const usersCollection = serversDB.collections.get('users'); const userRecord = await usersCollection.find(userId); const serverCollection = serversDB.collections.get('servers'); const serverRecord = await serverCollection.find(server); await serversDB.batch( userRecord.prepareDestroyPermanently(), serverRecord.prepareDestroyPermanently() ); }); } catch (error) { // Do nothing } await RNUserDefaults.clear('currentServer'); await RNUserDefaults.clear(TOKEN_KEY); await RNUserDefaults.clear(`${ TOKEN_KEY }-${ server }`); try { const db = database.active; await db.action(() => db.unsafeResetDatabase()); } catch (error) { console.log(error); } }, registerPushToken() { return new Promise(async(resolve) => { const token = getDeviceToken(); if (token) { const type = isIOS ? 'apn' : 'gcm'; const data = { value: token, type, appName: getBundleId }; try { // RC 0.60.0 await this.sdk.post('push.token', data); } catch (error) { console.log(error); } } return resolve(); }); }, removePushToken() { const token = getDeviceToken(); if (token) { // RC 0.60.0 return this.sdk.del('push.token', { token }); } return Promise.resolve(); }, loadMissedMessages, loadMessagesForRoom, loadThreadMessages, sendMessage, getRooms, readMessages, async resendMessage(message, tmid) { const db = database.active; try { await db.action(async() => { await message.update((m) => { m.status = messagesStatus.TEMP; }); }); let m = { id: message.id, msg: message.msg, subscription: { id: message.subscription.id } }; if (tmid) { m = { ...m, tmid }; } await sendMessageCall.call(this, m); } catch (e) { log(e); } }, async search({ text, filterUsers = true, filterRooms = true }) { const searchText = text.trim(); if (this.oldPromise) { this.oldPromise('cancel'); } if (searchText === '') { delete this.oldPromise; return []; } const db = database.active; let data = await db.collections.get('subscriptions').query( Q.where('name', Q.like(`%${ Q.sanitizeLikeString(searchText) }%`)) ).fetch(); if (filterUsers && !filterRooms) { data = data.filter(item => item.t === 'd'); } else if (!filterUsers && filterRooms) { data = data.filter(item => item.t !== 'd'); } data = data.slice(0, 7); const usernames = data.map(sub => sub.name); try { if (data.length < 7) { const { users, rooms } = await Promise.race([ RocketChat.spotlight(searchText, usernames, { users: filterUsers, rooms: filterRooms }), new Promise((resolve, reject) => this.oldPromise = reject) ]); data = data.concat(users.map(user => ({ ...user, rid: user.username, name: user.username, t: 'd', search: true })), rooms.map(room => ({ rid: room._id, ...room, search: true }))); } delete this.oldPromise; return data; } catch (e) { console.warn(e); return data; // return []; } }, spotlight(search, usernames, type) { // RC 0.51.0 return this.sdk.methodCall('spotlight', search, usernames, type); }, createDirectMessage(username) { // RC 0.59.0 return this.sdk.post('im.create', { username }); }, joinRoom(roomId, type) { // TODO: join code // RC 0.48.0 if (type === 'p') { return this.sdk.methodCall('joinRoom', roomId); } return this.sdk.post('channels.join', { roomId }); }, sendFileMessage, cancelUpload, isUploadActive, getSettings, setSettings, getPermissions, getCustomEmojis, setCustomEmojis, getSlashCommands, getRoles, parseSettings: settings => settings.reduce((ret, item) => { ret[item._id] = item[defaultSettings[item._id].type]; return ret; }, {}), _prepareSettings(settings) { return settings.map((setting) => { setting[defaultSettings[setting._id].type] = setting.value; return setting; }); }, deleteMessage(messageId, rid) { // RC 0.48.0 return this.sdk.post('chat.delete', { msgId: messageId, roomId: rid }); }, editMessage(message) { const { id, msg, rid } = message; // RC 0.49.0 return this.sdk.post('chat.update', { roomId: rid, msgId: id, text: msg }); }, toggleStarMessage(messageId, starred) { if (starred) { // RC 0.59.0 return this.sdk.post('chat.unStarMessage', { messageId }); } // RC 0.59.0 return this.sdk.post('chat.starMessage', { messageId }); }, togglePinMessage(messageId, pinned) { if (pinned) { // RC 0.59.0 return this.sdk.post('chat.unPinMessage', { messageId }); } // RC 0.59.0 return this.sdk.post('chat.pinMessage', { messageId }); }, reportMessage(messageId) { return this.sdk.post('chat.reportMessage', { messageId, description: 'Message reported by user' }); }, async getRoom(rid) { try { const db = database.active; const room = await db.collections.get('subscriptions').find(rid); return Promise.resolve(room); } catch (error) { return Promise.reject(new Error('Room not found')); } }, async getPermalinkMessage(message) { let room; try { room = await RocketChat.getRoom(message.subscription.id); } catch (e) { log(e); return null; } const { server } = reduxStore.getState().server; const roomType = { p: 'group', c: 'channel', d: 'direct' }[room.t]; return `${ server }/${ roomType }/${ room.name }?msg=${ message.id }`; }, getPermalinkChannel(channel) { const { server } = reduxStore.getState().server; const roomType = { p: 'group', c: 'channel', d: 'direct' }[channel.t]; return `${ server }/${ roomType }/${ channel.name }`; }, subscribe(...args) { return this.sdk.subscribe(...args); }, unsubscribe(subscription) { return this.sdk.unsubscribe(subscription); }, onStreamData(...args) { return this.sdk.onStreamData(...args); }, emitTyping(room, t = true) { const { login } = reduxStore.getState(); return this.sdk.methodCall('stream-notify-room', `${ room }/typing`, login.user.username, t); }, setUserPresenceAway() { return this.sdk.methodCall('UserPresence:away'); }, setUserPresenceOnline() { return this.sdk.methodCall('UserPresence:online'); }, setUserPresenceDefaultStatus(status) { return this.sdk.methodCall('UserPresence:setDefaultStatus', status); }, setReaction(emoji, messageId) { // RC 0.62.2 return this.sdk.post('chat.react', { emoji, messageId }); }, toggleFavorite(roomId, favorite) { // RC 0.64.0 return this.sdk.post('rooms.favorite', { roomId, favorite }); }, toggleRead(read, roomId) { if (read) { return this.sdk.post('subscriptions.unread', { roomId }); } return this.sdk.post('subscriptions.read', { rid: roomId }); }, getRoomMembers(rid, allUsers, skip = 0, limit = 10) { // RC 0.42.0 return this.sdk.methodCall('getUsersOfRoom', rid, allUsers, { skip, limit }); }, getUserRoles() { // RC 0.27.0 return this.sdk.methodCall('getUserRoles'); }, getRoomCounters(roomId, t) { // RC 0.65.0 return this.sdk.get(`${ this.roomTypeToApiType(t) }.counters`, { roomId }); }, getChannelInfo(roomId) { // RC 0.48.0 return this.sdk.get('channels.info', { roomId }); }, getUserInfo(userId) { // RC 0.48.0 return this.sdk.get('users.info', { userId }); }, getRoomInfo(roomId) { // RC 0.72.0 return this.sdk.get('rooms.info', { roomId }); }, getRoomMemberId(rid, currentUserId) { if (rid === `${ currentUserId }${ currentUserId }`) { return currentUserId; } return rid.replace(currentUserId, '').trim(); }, toggleBlockUser(rid, blocked, block) { if (block) { // RC 0.49.0 return this.sdk.methodCall('blockUser', { rid, blocked }); } // RC 0.49.0 return this.sdk.methodCall('unblockUser', { rid, blocked }); }, leaveRoom(roomId, t) { // RC 0.48.0 return this.sdk.post(`${ this.roomTypeToApiType(t) }.leave`, { roomId }); }, eraseRoom(roomId, t) { // RC 0.49.0 return this.sdk.post(`${ this.roomTypeToApiType(t) }.delete`, { roomId }); }, toggleMuteUserInRoom(rid, username, mute) { if (mute) { // RC 0.51.0 return this.sdk.methodCall('muteUserInRoom', { rid, username }); } // RC 0.51.0 return this.sdk.methodCall('unmuteUserInRoom', { rid, username }); }, toggleArchiveRoom(roomId, t, archive) { if (archive) { // RC 0.48.0 return this.sdk.post(`${ this.roomTypeToApiType(t) }.archive`, { roomId }); } // RC 0.48.0 return this.sdk.post(`${ this.roomTypeToApiType(t) }.unarchive`, { roomId }); }, hideRoom(roomId, t) { return this.sdk.post(`${ this.roomTypeToApiType(t) }.close`, { roomId }); }, saveRoomSettings(rid, params) { // RC 0.55.0 return this.sdk.methodCall('saveRoomSettings', rid, params); }, saveUserProfile(data, customFields) { // RC 0.62.2 return this.sdk.post('users.updateOwnBasicInfo', { data, customFields }); }, saveUserPreferences(params) { // RC 0.51.0 return this.sdk.methodCall('saveUserPreferences', params); }, saveNotificationSettings(roomId, notifications) { // RC 0.63.0 return this.sdk.post('rooms.saveNotification', { roomId, notifications }); }, addUsersToRoom(rid) { let { users } = reduxStore.getState().selectedUsers; users = users.map(u => u.name); // RC 0.51.0 return this.sdk.methodCall('addUsersToRoom', { rid, users }); }, getSingleMessage(msgId) { // RC 0.57.0 return this.sdk.methodCall('getSingleMessage', msgId); }, async hasPermission(permissions, rid) { const db = database.active; const subsCollection = db.collections.get('subscriptions'); const permissionsCollection = db.collections.get('permissions'); let roomRoles = []; try { // get the room from database const room = await subsCollection.find(rid); // get room roles roomRoles = room.roles; } catch (error) { console.log('hasPermission -> Room not found'); return permissions.reduce((result, permission) => { result[permission] = false; return result; }, {}); } // get permissions from database try { const permissionsFiltered = await permissionsCollection.query(Q.where('id', Q.oneOf(permissions))).fetch(); // get user roles on the server from redux const userRoles = (reduxStore.getState().login.user && reduxStore.getState().login.user.roles) || []; // merge both roles const mergedRoles = [...new Set([...roomRoles, ...userRoles])]; // return permissions in object format // e.g. { 'edit-room': true, 'set-readonly': false } return permissions.reduce((result, permission) => { result[permission] = false; const permissionFound = permissionsFiltered.find(p => p.id === permission); if (permissionFound) { result[permission] = returnAnArray(permissionFound.roles).some(r => mergedRoles.includes(r)); } return result; }, {}); } catch (e) { log(e); } }, getAvatarSuggestion() { // RC 0.51.0 return this.sdk.methodCall('getAvatarSuggestion'); }, resetAvatar(userId) { // RC 0.55.0 return this.sdk.post('users.resetAvatar', { userId }); }, setAvatarFromService({ data, contentType = '', service = null }) { // RC 0.51.0 return this.sdk.methodCall('setAvatarFromService', data, contentType, service); }, async getUseMarkdown() { const useMarkdown = await AsyncStorage.getItem(MARKDOWN_KEY); if (useMarkdown === null) { return true; } return JSON.parse(useMarkdown); }, async getAllowCrashReport() { const allowCrashReport = await AsyncStorage.getItem(CRASH_REPORT_KEY); if (allowCrashReport === null) { return true; } return JSON.parse(allowCrashReport); }, async getSortPreferences() { const prefs = await RNUserDefaults.objectForKey(SORT_PREFS_KEY); return prefs; }, async saveSortPreference(param) { try { let prefs = await RocketChat.getSortPreferences(); prefs = { ...prefs, ...param }; return await RNUserDefaults.setObjectForKey(SORT_PREFS_KEY, prefs); } catch (error) { console.warn(error); } }, async getLoginServices(server) { try { let loginServices = []; const loginServicesResult = await fetch(`${ server }/api/v1/settings.oauth`).then(response => response.json()); if (loginServicesResult.success && loginServicesResult.services.length > 0) { const { services } = loginServicesResult; loginServices = services; const loginServicesReducer = loginServices.reduce((ret, item) => { const name = item.name || item.buttonLabelText || item.service; const authType = this._determineAuthType(item); if (authType !== 'not_supported') { ret[name] = { ...item, name, authType }; } return ret; }, {}); reduxStore.dispatch(setLoginServices(loginServicesReducer)); } return Promise.resolve(loginServices.length); } catch (error) { console.warn(error); return Promise.reject(); } }, _determineAuthType(services) { const { name, custom, showButton = true, service } = services; if (custom && showButton) { return 'oauth_custom'; } if (service === 'saml') { return 'saml'; } if (service === 'cas') { return 'cas'; } // TODO: remove this after other oauth providers are implemented. e.g. Drupal, github_enterprise const availableOAuth = ['facebook', 'github', 'gitlab', 'google', 'linkedin', 'meteor-developer', 'twitter']; return availableOAuth.includes(name) ? 'oauth' : 'not_supported'; }, getUsernameSuggestion() { // RC 0.65.0 return this.sdk.get('users.getUsernameSuggestion'); }, roomTypeToApiType(t) { const types = { c: 'channels', d: 'im', p: 'groups' }; return types[t]; }, getFiles(roomId, type, offset) { // RC 0.59.0 return this.sdk.get(`${ this.roomTypeToApiType(type) }.files`, { roomId, offset, sort: { uploadedAt: -1 }, fields: { name: 1, description: 1, size: 1, type: 1, uploadedAt: 1, url: 1, userId: 1 } }); }, getMessages(roomId, type, query, offset) { // RC 0.59.0 return this.sdk.get(`${ this.roomTypeToApiType(type) }.messages`, { roomId, query, offset, sort: { ts: -1 } }); }, getReadReceipts(messageId) { return this.sdk.get('chat.getMessageReadReceipts', { messageId }); }, searchMessages(roomId, searchText) { // RC 0.60.0 return this.sdk.get('chat.search', { roomId, searchText }); }, toggleFollowMessage(mid, follow) { // RC 1.0 if (follow) { return this.sdk.post('chat.followMessage', { mid }); } return this.sdk.post('chat.unfollowMessage', { mid }); }, getThreadsList({ rid, count, offset }) { // RC 1.0 return this.sdk.get('chat.getThreadsList', { rid, count, offset, sort: { ts: -1 } }); }, getSyncThreadsList({ rid, updatedSince }) { // RC 1.0 return this.sdk.get('chat.syncThreadsList', { rid, updatedSince }); }, runSlashCommand(command, roomId, params) { // RC 0.60.2 return this.sdk.post('commands.run', { command, roomId, params }); }, getCommandPreview(command, roomId, params) { // RC 0.65.0 return this.sdk.get('commands.preview', { command, roomId, params }); }, executeCommandPreview(command, params, roomId, previewItem) { // RC 0.65.0 return this.sdk.post('commands.preview', { command, params, roomId, previewItem }); }, _setUser(ddpMessage) { this.activeUsers = this.activeUsers || {}; const { user } = reduxStore.getState().login; if (ddpMessage.fields && user && user.id === ddpMessage.id) { reduxStore.dispatch(setUser(ddpMessage.fields)); } if (ddpMessage.cleared && user && user.id === ddpMessage.id) { reduxStore.dispatch(setUser({ status: 'offline' })); } if (!this._setUserTimer) { this._setUserTimer = setTimeout(() => { const activeUsersBatch = this.activeUsers; InteractionManager.runAfterInteractions(() => { reduxStore.dispatch(setActiveUsers(activeUsersBatch)); }); this._setUserTimer = null; return this.activeUsers = {}; }, 10000); } if (!ddpMessage.fields) { this.activeUsers[ddpMessage.id] = 'offline'; } else if (ddpMessage.fields.status) { this.activeUsers[ddpMessage.id] = ddpMessage.fields.status; } }, getUserPresence() { return new Promise(async(resolve) => { const serverVersion = reduxStore.getState().server.version; // if server is lower than 1.1.0 if (serverVersion && semver.lt(semver.coerce(serverVersion), '1.1.0')) { if (this.activeUsersSubTimeout) { clearTimeout(this.activeUsersSubTimeout); this.activeUsersSubTimeout = false; } this.activeUsersSubTimeout = setTimeout(() => { this.sdk.subscribe('activeUsers'); }, 5000); return resolve(); } else { const params = {}; // if (this.lastUserPresenceFetch) { // params.from = this.lastUserPresenceFetch.toISOString(); // } // RC 1.1.0 const result = await this.sdk.get('users.presence', params); if (result.success) { const activeUsers = result.users.reduce((ret, item) => { ret[item._id] = item.status; return ret; }, {}); InteractionManager.runAfterInteractions(() => { reduxStore.dispatch(setActiveUsers(activeUsers)); }); this.sdk.subscribe('stream-notify-logged', 'user-status'); return resolve(); } } }); }, getDirectory({ query, count, offset, sort }) { // RC 1.0 return this.sdk.get('directory', { query, count, offset, sort }); }, async canAutoTranslate() { const db = database.active; try { const AutoTranslate_Enabled = reduxStore.getState().settings && reduxStore.getState().settings.AutoTranslate_Enabled; if (!AutoTranslate_Enabled) { return false; } const permissionsCollection = db.collections.get('permissions'); const autoTranslatePermission = await permissionsCollection.find('auto-translate'); const userRoles = (reduxStore.getState().login.user && reduxStore.getState().login.user.roles) || []; return autoTranslatePermission.roles.some(role => userRoles.includes(role)); } catch (e) { log(e); return false; } }, saveAutoTranslate({ rid, field, value, options }) { return this.sdk.methodCall('autoTranslate.saveSettings', rid, field, value, options); }, getSupportedLanguagesAutoTranslate() { return this.sdk.methodCall('autoTranslate.getSupportedLanguages', 'en'); }, translateMessage(message, targetLanguage) { return this.sdk.methodCall('autoTranslate.translateMessage', message, targetLanguage); }, getRoomTitle(room) { const { UI_Use_Real_Name: useRealName } = reduxStore.getState().settings; return ((room.prid || useRealName) && room.fname) || room.name; }, findOrCreateInvite({ rid, days, maxUses }) { // RC 2.4.0 return this.sdk.post('findOrCreateInvite', { rid, days, maxUses }); }, validateInviteToken(token) { // RC 2.4.0 return this.sdk.post('validateInviteToken', { token }); }, useInviteToken(token) { // RC 2.4.0 return this.sdk.post('useInviteToken', { token }); } }; export default RocketChat;