diff --git a/app/api/server/v1/rooms.js b/app/api/server/v1/rooms.js index 7cab7a10977b..c4892b409ad0 100644 --- a/app/api/server/v1/rooms.js +++ b/app/api/server/v1/rooms.js @@ -222,7 +222,7 @@ API.v1.addRoute('rooms.leave', { authRequired: true }, { API.v1.addRoute('rooms.createDiscussion', { authRequired: true }, { post() { - const { prid, pmid, reply, t_name, users } = this.bodyParams; + const { prid, pmid, reply, t_name, users, t } = this.bodyParams; if (!prid) { return API.v1.failure('Body parameter "prid" is required.'); } @@ -238,6 +238,7 @@ API.v1.addRoute('rooms.createDiscussion', { authRequired: true }, { pmid, t_name, reply, + t, users: users || [], })); diff --git a/app/articles/README.md b/app/articles/README.md new file mode 100644 index 000000000000..f117eed00aab --- /dev/null +++ b/app/articles/README.md @@ -0,0 +1 @@ +Readme file for articles. diff --git a/app/articles/client/index.js b/app/articles/client/index.js new file mode 100644 index 000000000000..e69de29bb2d1 diff --git a/app/articles/index.js b/app/articles/index.js new file mode 100644 index 000000000000..e69de29bb2d1 diff --git a/app/articles/server/api/api.js b/app/articles/server/api/api.js new file mode 100644 index 000000000000..933c510dc05e --- /dev/null +++ b/app/articles/server/api/api.js @@ -0,0 +1,104 @@ +import { Restivus } from 'meteor/nimble:restivus'; +import _ from 'underscore'; + +import { processWebhookMessage } from '../../../lib'; +import { API } from '../../../api'; +import { settings } from '../../../settings'; +import * as Models from '../../../models'; + +const Api = new Restivus({ + enableCors: true, + apiPath: 'ghooks/', + auth: { + user() { + const payloadKeys = Object.keys(this.bodyParams); + const payloadIsWrapped = (this.bodyParams && this.bodyParams.payload) && payloadKeys.length === 1; + if (payloadIsWrapped && this.request.headers['content-type'] === 'application/x-www-form-urlencoded') { + try { + this.bodyParams = JSON.parse(this.bodyParams.payload); + } catch ({ message }) { + return { + error: { + statusCode: 400, + body: { + success: false, + error: message, + }, + }, + }; + } + } + + this.announceToken = settings.get('Announcement_Token'); + const { blogId } = this.request.params; + const token = decodeURIComponent(this.request.params.token); + + if (this.announceToken !== `${ blogId }/${ token }`) { + return { + error: { + statusCode: 404, + body: { + success: false, + error: 'Invalid token provided.', + }, + }, + }; + } + + const user = this.bodyParams.userId ? Models.Users.findOne({ _id: this.bodyParams.userId }) : Models.Users.findOne({ username: this.bodyParams.username }); + + return { user }; + }, + }, +}); + +function executeAnnouncementRest() { + const defaultValues = { + channel: this.bodyParams.channel, + alias: this.bodyParams.alias, + avatar: this.bodyParams.avatar, + emoji: this.bodyParams.emoji, + }; + + // TODO: Turn this into an option on the integrations - no body means a success + // TODO: Temporary fix for https://github.com/RocketChat/Rocket.Chat/issues/7770 until the above is implemented + if (!this.bodyParams || (_.isEmpty(this.bodyParams) && !this.integration.scriptEnabled)) { + // return RocketChat.API.v1.failure('body-empty'); + return API.v1.success(); + } + + try { + const message = processWebhookMessage(this.bodyParams, this.user, defaultValues); + if (_.isEmpty(message)) { + return API.v1.failure('unknown-error'); + } + + return API.v1.success(); + } catch ({ error, message }) { + return API.v1.failure(error || message); + } +} + +function executefetchUserRest() { + try { + const { _id, name, username, emails } = this.user; + const user = { _id, name, username, emails }; + + return API.v1.success({ user }); + } catch ({ error, message }) { + return API.v1.failure(error || message); + } +} + +Api.addRoute(':blogId/:token', { authRequired: true }, { + post: executeAnnouncementRest, + get: executeAnnouncementRest, +}); + +// If a user is editor/admin in Ghost but is not an admin in RC, +// then the e-mail will not be provided to that user +// This method will allow user to fetch user with email. +Api.addRoute(':blogId/:token/getUser', { authRequired: true }, { + post: executefetchUserRest, + get: executefetchUserRest, +}); diff --git a/app/articles/server/index.js b/app/articles/server/index.js new file mode 100644 index 000000000000..b9804b9805f2 --- /dev/null +++ b/app/articles/server/index.js @@ -0,0 +1,6 @@ +import './settings'; +import './methods/admin'; +import './methods/user'; +import './api/api'; +import './lib/triggerHandler'; +import './triggers'; diff --git a/app/articles/server/lib/triggerHandler.js b/app/articles/server/lib/triggerHandler.js new file mode 100644 index 000000000000..6db1feeef5ad --- /dev/null +++ b/app/articles/server/lib/triggerHandler.js @@ -0,0 +1,150 @@ +import { Meteor } from 'meteor/meteor'; +import { HTTP } from 'meteor/http'; + +import { settings } from '../../../settings'; +import { API } from '../utils/url'; + +const api = new API(); + +export const triggerHandler = new class ArticlesSettingsHandler { + constructor() { + this.trigger = {}; + } + + eventNameArgumentsToObject(...args) { + const argObject = { + event: args[0], + }; + switch (argObject.event) { + case 'userEmail': + case 'userRealname': + case 'userAvatar': + case 'userName': + if (args.length >= 2) { + argObject.user = args[1]; + } + break; + case 'roomType': + case 'roomName': + if (args.length >= 2) { + argObject.room = args[1]; + } + break; + case 'siteTitle': + argObject.article = args[1]; + break; + default: + argObject.event = undefined; + break; + } + return argObject; + } + + mapEventArgsToData(data, { event, room, user, article }) { + data.event = event; + switch (event) { + case 'userEmail': + case 'userRealname': + case 'userAvatar': + case 'userName': + data.user_id = user._id; + + if (user.avatar) { + data.avatar = user.avatar; + } + + if (user.name) { + data.name = user.name; + } + + if (user.email) { + data.email = user.email; + } + + if (user.username) { + data.username = user.username; + } + break; + case 'roomType': + case 'roomName': + data.room_id = room.rid; + + if (room.name) { + data.name = room.name; + } + + if (room.type) { + data.type = room.type; + } + break; + case 'siteTitle': + if (article && article.title) { + data.title = article.title; + } + break; + default: + break; + } + } + + executeTrigger(...args) { + const argObject = this.eventNameArgumentsToObject(...args); + const { event } = argObject; + + if (!event) { + return; + } + + if (settings.get('Articles_enabled')) { + const token = settings.get('Settings_Token'); + this.trigger.api = api.rhooks(token); + this.trigger.retryCount = 5; + } + + this.executeTriggerUrl(argObject, 0); + } + + executeTriggerUrl({ event, room, user, article }, tries = 0) { + if (!this.trigger.api) { + return; + } + const url = this.trigger.api; + + const data = {}; + + this.mapEventArgsToData(data, { event, room, user, article }); + + const opts = { + params: {}, + method: 'POST', + url, + data, + auth: undefined, + npmRequestOptions: { + rejectUnauthorized: !settings.get('Allow_Invalid_SelfSigned_Certs'), + strictSSL: !settings.get('Allow_Invalid_SelfSigned_Certs'), + }, + headers: { + 'User-Agent': 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/41.0.2227.0 Safari/537.36', + }, + }; + + if (!opts.url || !opts.method) { + return; + } + + HTTP.call(opts.method, opts.url, opts, (error, result) => { + // if the result contained nothing or wasn't a successful statusCode + if (!result) { + if (tries < this.trigger.retryCount) { + // 2 seconds, 4 seconds, 8 seconds + const waitTime = Math.pow(2, tries + 1) * 1000; + + Meteor.setTimeout(() => { + this.executeTriggerUrl({ event, room, user }, tries + 1); + }, waitTime); + } + } + }); + } +}(); diff --git a/app/articles/server/logoutCleanUp.js b/app/articles/server/logoutCleanUp.js new file mode 100644 index 000000000000..9c527585ac91 --- /dev/null +++ b/app/articles/server/logoutCleanUp.js @@ -0,0 +1,19 @@ +import { Meteor } from 'meteor/meteor'; +import { HTTP } from 'meteor/http'; + +import { settings } from '../../settings'; +import { API } from './utils/url'; + +const api = new API(); + +export function ghostCleanUp(cookie) { + const rcUrl = Meteor.absoluteUrl().replace(/\/$/, ''); + try { + if (settings.get('Articles_enabled')) { + HTTP.call('DELETE', api.session(), { headers: { cookie, referer: rcUrl } }); + } + } catch (e) { + // Do nothing if failed to logout from Ghost. + // Error will be because user has not logged in to Ghost. + } +} diff --git a/app/articles/server/methods/admin.js b/app/articles/server/methods/admin.js new file mode 100644 index 000000000000..842cd112def1 --- /dev/null +++ b/app/articles/server/methods/admin.js @@ -0,0 +1,77 @@ +import { Meteor } from 'meteor/meteor'; +import { HTTP } from 'meteor/http'; +import _ from 'underscore'; +import { Random } from 'meteor/random'; + +import { API } from '../utils/url'; +import { settings } from '../../../settings'; + +const api = new API(); + +// Try to get a verified email, if available. +function getVerifiedEmail(emails) { + const email = _.find(emails, (e) => e.verified); + return email || emails[0].address; +} + +function setupGhost(user, token) { + const rcUrl = Meteor.absoluteUrl().replace(/\/$/, ''); + const blogTitle = settings.get('Article_Site_title'); + const blogToken = Random.id(17); + const announceToken = `${ blogToken }/${ Random.id(24) }`; + const settingsToken = `${ blogToken }/${ Random.id(24) }`; + settings.updateById('Announcement_Token', announceToken); + settings.updateById('Settings_Token', settingsToken); + const data = { + setup: [{ + rc_url: rcUrl, + rc_id: user._id, + rc_token: token, + name: user.name, + email: getVerifiedEmail(user.emails), + announce_token: announceToken, + settings_token: settingsToken, + blogTitle, + }], + }; + return HTTP.call('POST', api.setup(), { data, headers: { 'Content-Type': 'application/json' } }); +} + +function redirectGhost() { + return { + link: api.siteUrl(), + message: 'Ghost is Set up. Redirecting.', + }; +} + +Meteor.methods({ + Articles_admin_panel(token) { + const enabled = settings.get('Articles_enabled'); + + if (!enabled) { + throw new Meteor.Error('Articles are disabled'); + } + const user = Meteor.users.findOne(Meteor.userId()); + + try { + let response = HTTP.call('GET', api.setup()); + + if (response.data && response.data.setup && response.data.setup[0]) { + if (response.data.setup[0].status) { // Ghost site is already setup + return redirectGhost(); + } // Setup Ghost Site and set title + response = setupGhost(user, token); + if (response.statusCode === 201 && response.content) { + return redirectGhost(); + } if (response.errors) { + throw new Meteor.Error(response.errors.message || 'Unable to setup. Make sure Ghost is running'); + } + } else { + throw new Meteor.Error('Unable to redirect. Make sure Ghost is running.'); + } + } catch (e) { + console.log(e); + throw new Meteor.Error(e.error || 'Unable to connect to Ghost. Make sure Ghost is running.'); + } + }, +}); diff --git a/app/articles/server/methods/user.js b/app/articles/server/methods/user.js new file mode 100644 index 000000000000..414502ef2068 --- /dev/null +++ b/app/articles/server/methods/user.js @@ -0,0 +1,88 @@ +import { Meteor } from 'meteor/meteor'; +import { HTTP } from 'meteor/http'; + +import { API } from '../utils/url'; +import { settings } from '../../../settings'; + +const api = new API(); + +function addUser(user, accessToken) { + const data = { + user: [{ + rc_username: user.username, + role: 'Author', // User can add itself as Author, even if he/she is admin in RC + rc_uid: user._id, + rc_token: accessToken, + }], + }; + return HTTP.call('POST', api.createAccount(), { data, headers: { 'Content-Type': 'application/json' } }); +} + +function userExist(user, accessToken) { + const data = { + user: [{ + rc_uid: user._id, + rc_token: accessToken, + }], + }; + const response = HTTP.call('GET', api.userExist(), { data, headers: { 'Content-Type': 'application/json' } }); + return response.data && response.data.users[0] && response.data.users[0].exist; +} + +function inviteSetting() { + const response = HTTP.call('GET', api.invite()); + const { settings } = response.data; + + if (settings && settings[0] && settings[0].key === 'invite_only') { + return settings[0].value; + } + // default value in Ghost + return false; +} + +function redirectGhost() { + return { + link: api.siteUrl(), + message: 'Ghost is Set up. Redirecting.', + }; +} + +Meteor.methods({ + redirectUserToArticles(accessToken) { + const enabled = settings.get('Articles_enabled'); + + if (!enabled) { + throw new Meteor.Error('Articles are disabled'); + } + const user = Meteor.users.findOne(Meteor.userId()); + + try { + let response = HTTP.call('GET', api.setup()); + if (response.data && response.data.setup && response.data.setup[0]) { + if (response.data.setup[0].status) { // Ghost site is already setup + const exist = userExist(user, accessToken); + if (exist) { + return redirectGhost(); + } + const inviteOnly = inviteSetting(); + + if (!inviteOnly) { + response = addUser(user, accessToken); + if (response.statusCode === 200) { + return redirectGhost(); + } + throw new Meteor.Error('Unable to setup your account.'); + } else { + throw new Meteor.Error('You are not a member of Ghost. Ask admin to add'); + } + } else { // Cannot setup Ghost from sidenav + throw new Meteor.Error('Ghost is not set up. Setup can be done from Admin Panel.'); + } + } else { + throw new Meteor.Error('Unable to redirect.'); + } + } catch (e) { + throw new Meteor.Error(e.error || 'Unable to connect to Ghost.'); + } + }, +}); diff --git a/app/articles/server/settings.js b/app/articles/server/settings.js new file mode 100644 index 000000000000..273d978f4e94 --- /dev/null +++ b/app/articles/server/settings.js @@ -0,0 +1,64 @@ +import { Meteor } from 'meteor/meteor'; + +import { settings } from '../../settings'; + +const defaults = { + enable: false, +}; + +Meteor.startup(() => { + settings.addGroup('Articles', function() { + this.add('Articles_enabled', defaults.enable, { + type: 'boolean', + i18nLabel: 'Enable', + public: true, + }); + + this.add('Article_Site_title', 'Rocket.Chat', { + type: 'string', + enableQuery: { + _id: 'Articles_enabled', + value: true, + }, + public: true, + }); + + this.add('Article_Site_Url', 'http://localhost:2368', { + type: 'string', + enableQuery: { + _id: 'Articles_enabled', + value: true, + }, + public: true, + }); + + this.add('Announcement_Token', 'announcement_token', { + type: 'string', + readonly: true, + enableQuery: { + _id: 'Articles_enabled', + value: true, + }, + public: true, + }); + + this.add('Settings_Token', 'articles_settings_token', { + type: 'string', + readonly: true, + enableQuery: { + _id: 'Articles_enabled', + value: true, + }, + public: true, + }); + + this.add('Articles_admin_panel', 'Articles_admin_panel', { + type: 'link', + enableQuery: { + _id: 'Articles_enabled', + value: true, + }, + linkText: 'Article_Admin_Panel', + }); + }); +}); diff --git a/app/articles/server/triggers.js b/app/articles/server/triggers.js new file mode 100644 index 000000000000..4c726b3833a0 --- /dev/null +++ b/app/articles/server/triggers.js @@ -0,0 +1,22 @@ +import { callbacks } from '../../callbacks'; +import { triggerHandler } from './lib/triggerHandler'; +import { settings } from '../../settings'; + +const callbackHandler = function _callbackHandler(eventType) { + return function _wrapperFunction(...args) { + return triggerHandler.executeTrigger(eventType, ...args); + }; +}; + +const priority = settings.get('Articles_enabled') ? callbacks.priority.HIGH : callbacks.priority.LOW; + +callbacks.add('afterUserEmailChange', callbackHandler('userEmail'), priority); +callbacks.add('afterUserRealNameChange', callbackHandler('userRealname'), priority); +callbacks.add('afterUserAvatarChange', callbackHandler('userAvatar'), priority); +callbacks.add('afterUsernameChange', callbackHandler('userName'), priority); +callbacks.add('afterRoomTypeChange', callbackHandler('roomType'), priority); +callbacks.add('afterRoomNameChange', callbackHandler('roomName'), priority); + +settings.get('Article_Site_title', (key, value) => { + triggerHandler.executeTrigger('siteTitle', { title: value }); +}); diff --git a/app/articles/server/utils/url.js b/app/articles/server/utils/url.js new file mode 100644 index 000000000000..0eaef9d02cba --- /dev/null +++ b/app/articles/server/utils/url.js @@ -0,0 +1,42 @@ +import { settings } from '../../../settings'; + +export class API { + constructor() { + this.adminApi = '/ghost/api/v2/admin'; + } + + buildAPIUrl(type, subtype = '') { + const base = settings.get('Article_Site_Url').replace(/\/$/, ''); + const dir = `/${ type }/${ subtype }`; + return base + this.adminApi + dir; + } + + siteUrl() { + const base = settings.get('Article_Site_Url').replace(/\/$/, ''); + return `${ base }/ghost`; + } + + setup() { + return this.buildAPIUrl('authentication', 'setup'); + } + + session() { + return this.buildAPIUrl('session'); + } + + rhooks(token) { + return this.buildAPIUrl('rhooks', token); + } + + invite() { + return this.buildAPIUrl('invitesetting'); + } + + createAccount() { + return this.buildAPIUrl('authentication', 'adduser'); + } + + userExist() { + return this.buildAPIUrl('userexist'); + } +} diff --git a/app/channel-settings/server/functions/saveRoomType.js b/app/channel-settings/server/functions/saveRoomType.js index 9bf6cb4503ba..6831e239e9d8 100644 --- a/app/channel-settings/server/functions/saveRoomType.js +++ b/app/channel-settings/server/functions/saveRoomType.js @@ -2,6 +2,7 @@ import { Meteor } from 'meteor/meteor'; import { Match } from 'meteor/check'; import { TAPi18n } from 'meteor/tap:i18n'; +import { callbacks } from '../../../callbacks'; import { Rooms, Subscriptions, Messages } from '../../../models'; import { settings } from '../../../settings'; @@ -43,5 +44,6 @@ export const saveRoomType = function(rid, roomType, user, sendMessage = true) { } Messages.createRoomSettingsChangedWithTypeRoomIdMessageAndUser('room_changed_privacy', rid, message, user); } + callbacks.run('afterRoomTypeChange', { rid, type: roomType }); return result; }; diff --git a/app/discussion/server/methods/createDiscussion.js b/app/discussion/server/methods/createDiscussion.js index 78535311ad9d..63cd7470483d 100644 --- a/app/discussion/server/methods/createDiscussion.js +++ b/app/discussion/server/methods/createDiscussion.js @@ -33,7 +33,7 @@ const mentionMessage = (rid, { _id, username, name }, message_embedded) => { return Messages.insert(welcomeMessage); }; -const create = ({ prid, pmid, t_name, reply, users }) => { +const create = ({ prid, pmid, t_name, reply, t, users }) => { // if you set both, prid and pmid, and the rooms doesnt match... should throw an error) let message = false; if (pmid) { @@ -86,8 +86,10 @@ const create = ({ prid, pmid, t_name, reply, users }) => { // auto invite the replied message owner const invitedUsers = message ? [message.u.username, ...users] : users; - // discussions are always created as private groups - const discussion = createRoom('p', name, user.username, [...new Set(invitedUsers)], false, { + // discussions are created as private groups, if t is not given as 'c' + const type = t === 'c' ? 'c' : 'p'; + + const discussion = createRoom(type, name, user.username, [...new Set(invitedUsers)], false, { fname: t_name, description: message.msg, // TODO discussions remove topic: p_room.name, // TODO discussions remove @@ -121,7 +123,7 @@ Meteor.methods({ * @param {string} t_name - discussion name * @param {string[]} users - users to be added */ - createDiscussion({ prid, pmid, t_name, reply, users }) { + createDiscussion({ prid, pmid, t_name, reply, t, users }) { if (!settings.get('Discussion_enabled')) { throw new Meteor.Error('error-action-not-allowed', 'You are not allowed to create a discussion', { method: 'createDiscussion' }); } @@ -135,6 +137,6 @@ Meteor.methods({ throw new Meteor.Error('error-action-not-allowed', 'You are not allowed to create a discussion', { method: 'createDiscussion' }); } - return create({ uid, prid, pmid, t_name, reply, users }); + return create({ uid, prid, pmid, t_name, reply, t, users }); }, }); diff --git a/app/file-upload/client/lib/fileUploadHandler.js b/app/file-upload/client/lib/fileUploadHandler.js index fe13fdcbb965..a93c8e5c25aa 100644 --- a/app/file-upload/client/lib/fileUploadHandler.js +++ b/app/file-upload/client/lib/fileUploadHandler.js @@ -27,7 +27,11 @@ export const fileUploadHandler = (directive, meta, file) => { Tracker.autorun(function() { if (Meteor.userId()) { - document.cookie = `rc_uid=${ escape(Meteor.userId()) }; path=/`; - document.cookie = `rc_token=${ escape(Accounts._storedLoginToken()) }; path=/`; + let domain = Meteor.absoluteUrl().replace(/.*\/\//, ''); + domain = domain.replace(/:.*\/$/, ''); + domain = domain.replace(/\/$/, ''); + domain = `.${ domain }`; + document.cookie = `rc_uid=${ escape(Meteor.userId()) }; domain=${ domain }; path=/`; + document.cookie = `rc_token=${ escape(Accounts._storedLoginToken()) }; domain=${ domain }; path=/`; } }); diff --git a/app/lib/server/functions/setEmail.js b/app/lib/server/functions/setEmail.js index 982194ae78e5..4a06a5622d3b 100644 --- a/app/lib/server/functions/setEmail.js +++ b/app/lib/server/functions/setEmail.js @@ -4,6 +4,7 @@ import s from 'underscore.string'; import { Users } from '../../../models'; import { hasPermission } from '../../../authorization'; import { RateLimiter, validateEmailDomain } from '../lib'; +import { callbacks } from '../../../callbacks'; import { checkEmailAvailability } from '.'; @@ -21,7 +22,7 @@ const _setEmail = function(userId, email, shouldSendVerificationEmail = true) { const user = Users.findOneById(userId); - // User already has desired username, return + // User already has desired email, return if (user.emails && user.emails[0] && user.emails[0].address === email) { return user; } @@ -37,6 +38,7 @@ const _setEmail = function(userId, email, shouldSendVerificationEmail = true) { if (shouldSendVerificationEmail === true) { Meteor.call('sendConfirmationEmail', user.email); } + callbacks.run('afterUserEmailChange', { _id: user._id, email }); return user; }; diff --git a/app/lib/server/functions/setRealName.js b/app/lib/server/functions/setRealName.js index 40262fd612bf..23efbc52f085 100644 --- a/app/lib/server/functions/setRealName.js +++ b/app/lib/server/functions/setRealName.js @@ -6,6 +6,7 @@ import { settings } from '../../../settings'; import { Notifications } from '../../../notifications'; import { hasPermission } from '../../../authorization'; import { RateLimiter } from '../lib'; +import { callbacks } from '../../../callbacks'; export const _setRealName = function(userId, name) { name = s.trim(name); @@ -41,6 +42,7 @@ export const _setRealName = function(userId, name) { username: user.username, }); } + callbacks.run('afterUserRealNameChange', { _id: user._id, name }); return user; }; diff --git a/app/lib/server/functions/setUserAvatar.js b/app/lib/server/functions/setUserAvatar.js index 4e7ec6201c9b..82e1daaba497 100644 --- a/app/lib/server/functions/setUserAvatar.js +++ b/app/lib/server/functions/setUserAvatar.js @@ -5,6 +5,7 @@ import { RocketChatFile } from '../../../file'; import { FileUpload } from '../../../file-upload'; import { Users } from '../../../models'; import { Notifications } from '../../../notifications'; +import { callbacks } from '../../../callbacks'; export const setUserAvatar = function(user, dataURI, contentType, service) { let encoding; @@ -63,4 +64,5 @@ export const setUserAvatar = function(user, dataURI, contentType, service) { Notifications.notifyLogged('updateAvatar', { username: user.username }); }, 500); }); + callbacks.run('afterUserAvatarChange', { _id: user._id, avatar: image }); }; diff --git a/app/lib/server/functions/setUsername.js b/app/lib/server/functions/setUsername.js index 24e58416102a..cf879f0b454d 100644 --- a/app/lib/server/functions/setUsername.js +++ b/app/lib/server/functions/setUsername.js @@ -8,6 +8,7 @@ import { Users, Messages, Subscriptions, Rooms, LivechatDepartmentAgents } from import { hasPermission } from '../../../authorization'; import { RateLimiter } from '../lib'; import { Notifications } from '../../../notifications/server'; +import { callbacks } from '../../../callbacks'; import { checkUsernameAvailability, setUserAvatar, getAvatarSuggestionForUser } from '.'; @@ -95,6 +96,7 @@ export const _setUsername = function(userId, u) { name: user.name, username: user.username, }); + callbacks.run('afterUsernameChange', { _id: user._id, username }); return user; }; diff --git a/app/ui-admin/client/admin.html b/app/ui-admin/client/admin.html index 1eaa5c0c81a0..25d7118ba457 100644 --- a/app/ui-admin/client/admin.html +++ b/app/ui-admin/client/admin.html @@ -153,6 +153,14 @@ {{/if}} {{/if}} + {{#if $eq type 'link'}} + {{#if hasChanges section}} + {{_ "Save_to_enable_this_action"}} + {{else}} + + {{/if}} + {{/if}} + {{#if $eq type 'asset'}} {{#if value.url}}