import crypto from 'crypto'; import { Meteor } from 'meteor/meteor'; import { check } from 'meteor/check'; import { TAPi18n } from 'meteor/rocketchat:tap-i18n'; import { EJSON } from 'meteor/ejson'; import { DDPRateLimiter } from 'meteor/ddp-rate-limiter'; import s from 'underscore.string'; import { hasRole, hasPermission } from '../../../authorization/server'; import { Info } from '../../../utils/server'; import { Users } from '../../../models/server'; import { settings } from '../../../settings/server'; import { API } from '../api'; import { getDefaultUserFields } from '../../../utils/server/functions/getDefaultUserFields'; import { getURL } from '../../../utils/lib/getURL'; import { StdOut } from '../../../logger/server/streamer'; // DEPRECATED // Will be removed after v3.0.0 API.v1.addRoute('info', { authRequired: false }, { get() { const warningMessage = 'The endpoint "/v1/info" is deprecated and will be removed after version v3.0.0'; console.warn(warningMessage); const user = this.getLoggedInUser(); if (user && hasRole(user._id, 'admin')) { return API.v1.success(this.deprecationWarning({ endpoint: 'info', versionWillBeRemoved: '3.0.0', response: { info: Info, }, })); } return API.v1.success(this.deprecationWarning({ endpoint: 'info', versionWillBeRemoved: '3.0.0', response: { info: { version: Info.version, }, }, })); }, }); API.v1.addRoute('me', { authRequired: true }, { get() { return API.v1.success(this.getUserInfo(Users.findOneById(this.userId, { fields: getDefaultUserFields() }))); }, }); let onlineCache = 0; let onlineCacheDate = 0; const cacheInvalid = 60000; // 1 minute API.v1.addRoute('shield.svg', { authRequired: false, rateLimiterOptions: { numRequestsAllowed: 60, intervalTimeInMS: 60000 } }, { get() { const { type, icon } = this.queryParams; let { channel, name } = this.queryParams; if (!settings.get('API_Enable_Shields')) { throw new Meteor.Error('error-endpoint-disabled', 'This endpoint is disabled', { route: '/api/v1/shield.svg' }); } const types = settings.get('API_Shield_Types'); if (type && (types !== '*' && !types.split(',').map((t) => t.trim()).includes(type))) { throw new Meteor.Error('error-shield-disabled', 'This shield type is disabled', { route: '/api/v1/shield.svg' }); } const hideIcon = icon === 'false'; if (hideIcon && (!name || !name.trim())) { return API.v1.failure('Name cannot be empty when icon is hidden'); } let text; let backgroundColor = '#4c1'; switch (type) { case 'online': if (Date.now() - onlineCacheDate > cacheInvalid) { onlineCache = Users.findUsersNotOffline().count(); onlineCacheDate = Date.now(); } text = `${ onlineCache } ${ TAPi18n.__('Online') }`; break; case 'channel': if (!channel) { return API.v1.failure('Shield channel is required for type "channel"'); } text = `#${ channel }`; break; case 'user': if (settings.get('API_Shield_user_require_auth') && !this.getLoggedInUser()) { return API.v1.failure('You must be logged in to do this.'); } const user = this.getUserFromParams(); // Respect the server's choice for using their real names or not if (user.name && settings.get('UI_Use_Real_Name')) { text = `${ user.name }`; } else { text = `@${ user.username }`; } switch (user.status) { case 'online': backgroundColor = '#1fb31f'; break; case 'away': backgroundColor = '#dc9b01'; break; case 'busy': backgroundColor = '#bc2031'; break; case 'offline': backgroundColor = '#a5a1a1'; } break; default: text = TAPi18n.__('Join_Chat').toUpperCase(); } const iconSize = hideIcon ? 7 : 24; const leftSize = name ? name.length * 6 + 7 + iconSize : iconSize; const rightSize = text.length * 6 + 20; const width = leftSize + rightSize; const height = 20; channel = s.escapeHTML(channel); text = s.escapeHTML(text); name = s.escapeHTML(name); return { headers: { 'Content-Type': 'image/svg+xml;charset=utf-8' }, body: ` ${ hideIcon ? '' : `` } ${ name ? `${ name } ${ name }` : '' } ${ text } ${ text } `.trim().replace(/\>[\s]+\<'), }; }, }); API.v1.addRoute('spotlight', { authRequired: true }, { get() { check(this.queryParams, { query: String, }); const { query } = this.queryParams; const result = Meteor.runAsUser(this.userId, () => Meteor.call('spotlight', query), ); return API.v1.success(result); }, }); API.v1.addRoute('directory', { authRequired: true }, { get() { const { offset, count } = this.getPaginationItems(); const { sort, query } = this.parseJsonQuery(); const { text, type, workspace = 'local' } = query; if (sort && Object.keys(sort).length > 1) { return API.v1.failure('This method support only one "sort" parameter'); } const sortBy = sort ? Object.keys(sort)[0] : undefined; const sortDirection = sort && Object.values(sort)[0] === 1 ? 'asc' : 'desc'; const result = Meteor.runAsUser(this.userId, () => Meteor.call('browseChannels', { text, type, workspace, sortBy, sortDirection, offset: Math.max(0, offset), limit: Math.max(0, count), })); if (!result) { return API.v1.failure('Please verify the parameters'); } return API.v1.success({ result: result.results, count: result.results.length, offset, total: result.total, }); }, }); API.v1.addRoute('stdout.queue', { authRequired: true }, { get() { if (!hasPermission(this.userId, 'view-logs')) { return API.v1.unauthorized(); } return API.v1.success({ queue: StdOut.queue }); }, }); const mountResult = ({ id, error, result }) => ({ message: EJSON.stringify({ msg: 'result', id, error, result, }), }); const methodCall = () => ({ post() { check(this.bodyParams, { message: String, }); const { method, params, id } = EJSON.parse(this.bodyParams.message); const connectionId = this.token || crypto.createHash('md5').update(this.requestIp + this.request.headers['user-agent']).digest('hex'); const rateLimiterInput = { userId: this.userId, clientAddress: this.requestIp, type: 'method', name: method, connectionId, }; try { DDPRateLimiter._increment(rateLimiterInput); const rateLimitResult = DDPRateLimiter._check(rateLimiterInput); if (!rateLimitResult.allowed) { throw new Meteor.Error( 'too-many-requests', DDPRateLimiter.getErrorMessage(rateLimitResult), { timeToReset: rateLimitResult.timeToReset }, ); } const result = Meteor.call(method, ...params); return API.v1.success(mountResult({ id, result })); } catch (error) { return API.v1.success(mountResult({ id, error })); } }, }); // had to create two different endpoints for authenticated and non-authenticated calls // because restivus does not provide 'this.userId' if 'authRequired: false' API.v1.addRoute('method.call/:method', { authRequired: true, rateLimiterOptions: false }, methodCall()); API.v1.addRoute('method.callAnon/:method', { authRequired: false, rateLimiterOptions: false }, methodCall());