Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions .changeset/twelve-cougars-suffer.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
---
"@rocket.chat/meteor": patch
"@rocket.chat/rest-typings": patch
---

Adds deprecation warning on `livechat:saveCustomField` with new endpoint replacing it; `livechat/custom-fields.save`
87 changes: 67 additions & 20 deletions apps/meteor/app/livechat/server/api/v1/customField.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,11 @@ import {
isPOSTLivechatCustomFieldParams,
isPOSTLivechatCustomFieldsParams,
isPOSTLivechatRemoveCustomFields,
isPOSTLivechatSaveCustomFieldsParams,
POSTLivechatRemoveCustomFieldSuccess,
POSTLivechatSaveCustomFieldSuccess,
validateBadRequestErrorResponse,
validateForbiddenErrorResponse,
validateUnauthorizedErrorResponse,
} from '@rocket.chat/rest-typings';

Expand Down Expand Up @@ -91,29 +94,73 @@ API.v1.addRoute(
},
);

const livechatCustomFieldsEndpoints = API.v1.post(
'livechat/custom-fields.delete',
{
response: {
200: POSTLivechatRemoveCustomFieldSuccess,
400: validateBadRequestErrorResponse,
401: validateUnauthorizedErrorResponse,
const livechatCustomFieldsEndpoints = API.v1
.post(
'livechat/custom-fields.save',
{
response: {
200: POSTLivechatSaveCustomFieldSuccess,
400: validateBadRequestErrorResponse,
401: validateUnauthorizedErrorResponse,
403: validateForbiddenErrorResponse,
},
authRequired: true,
permissionsRequired: ['view-livechat-manager'], // is this permission appropriate for the targeted action?
body: isPOSTLivechatSaveCustomFieldsParams,
},
authRequired: true,
permissionsRequired: ['view-livechat-manager'], // is this permission appropriate for the targeted action?
body: isPOSTLivechatRemoveCustomFields,
},
async function action() {
const { customFieldId } = this.bodyParams;
async function action() {
const { customFieldId, customFieldData } = this.bodyParams;

const result = await LivechatCustomField.removeById(customFieldId);
if (result.deletedCount === 0) {
return API.v1.failure('Custom field not found');
}
if (!/^[0-9a-zA-Z-_]+$/.test(customFieldId)) {
return API.v1.failure('Invalid custom field name. Use only letters, numbers, hyphens and underscores.');
}

return API.v1.success();
},
);
if (customFieldId) {
const customField = await LivechatCustomField.findOneById(customFieldId);
if (!customField) {
return API.v1.failure('Custom Field Not found');
}
}

if (!customFieldId) {
const customField = await LivechatCustomField.findOneById(customFieldData.field);
if (customField) {
return API.v1.failure('Custom Field already exists');
}
}

const { field, label, scope, visibility, ...extraData } = customFieldData;
const result = await LivechatCustomField.createOrUpdateCustomField(customFieldId, field, label, scope, visibility, {
...extraData,
});

return API.v1.success({ customField: result });
},
)
.post(
'livechat/custom-fields.delete',
{
response: {
200: POSTLivechatRemoveCustomFieldSuccess,
400: validateBadRequestErrorResponse,
401: validateUnauthorizedErrorResponse,
403: validateForbiddenErrorResponse,
},
authRequired: true,
permissionsRequired: ['view-livechat-manager'], // is this permission appropriate for the targeted action?
body: isPOSTLivechatRemoveCustomFields,
},
async function action() {
const { customFieldId } = this.bodyParams;

const result = await LivechatCustomField.removeById(customFieldId);
if (result.deletedCount === 0) {
return API.v1.failure('Custom field not found');
}

return API.v1.success();
},
);

type LivechatCustomFieldsEndpoints = ExtractRoutesFromAPI<typeof livechatCustomFieldsEndpoints>;

Expand Down
2 changes: 2 additions & 0 deletions apps/meteor/app/livechat/server/methods/saveCustomField.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import { Match, check } from 'meteor/check';
import { Meteor } from 'meteor/meteor';

import { hasPermissionAsync } from '../../../authorization/server/functions/hasPermission';
import { methodDeprecationLogger } from '../../../lib/server/lib/deprecationWarningLogger';

declare module '@rocket.chat/ddp-client' {
// eslint-disable-next-line @typescript-eslint/naming-convention
Expand All @@ -25,6 +26,7 @@ declare module '@rocket.chat/ddp-client' {

Meteor.methods<ServerMethods>({
async 'livechat:saveCustomField'(_id, customFieldData) {
methodDeprecationLogger.method('livechat:saveCustomField', '8.0.0', '/v1/livechat/custom-fields.save');
const uid = Meteor.userId();
if (!uid || !(await hasPermissionAsync(uid, 'view-livechat-manager'))) {
throw new Meteor.Error('error-not-allowed', 'Not allowed', {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ import {
Box,
} from '@rocket.chat/fuselage';
import { useEffectEvent } from '@rocket.chat/fuselage-hooks';
import { useToastMessageDispatch, useMethod, useTranslation } from '@rocket.chat/ui-contexts';
import { useToastMessageDispatch, useTranslation, useEndpoint } from '@rocket.chat/ui-contexts';
import { useQueryClient } from '@tanstack/react-query';
import { useId, useMemo } from 'react';
import { FormProvider, useForm, Controller } from 'react-hook-form';
Expand Down Expand Up @@ -72,13 +72,16 @@ const EditCustomFields = ({ customFieldData, onClose }: { customFieldData?: Seri
formState: { isDirty, errors },
} = methods;

const saveCustomField = useMethod('livechat:saveCustomField');
const saveCustomField = useEndpoint('POST', '/v1/livechat/custom-fields.save');

const handleSave = useEffectEvent(async ({ visibility, ...data }: EditCustomFieldsFormData) => {
try {
await saveCustomField(customFieldData?._id as unknown as string, {
visibility: visibility ? 'visible' : 'hidden',
...data,
await saveCustomField({
customFieldId: customFieldData?._id as unknown as string,
customFieldData: {
visibility: visibility ? 'visible' : 'hidden',
...data,
},
});

dispatchToastMessage({ type: 'success', message: t('Saved') });
Expand Down
51 changes: 20 additions & 31 deletions apps/meteor/tests/e2e/utils/omnichannel/custom-field.ts
Original file line number Diff line number Diff line change
@@ -1,50 +1,39 @@
import type { ILivechatCustomField } from '@rocket.chat/core-typings';

import { parseMeteorResponse } from '../parseMeteorResponse';
import type { BaseTest } from '../test';

type CustomField = Omit<ILivechatCustomField, '_id' | '_updatedAt'> & { field: string };

export const removeCustomField = (api: BaseTest['api'], id: string) => {
return api.post('/method.call/livechat:deleteCustomField', {
method: 'livechat:saveCustomField',
params: [id],
id: 'id',
msg: 'method',
return api.post('/livechat/custom-fields.delete', {
customFieldId: id,
});
};

export const createCustomField = async (api: BaseTest['api'], overwrides: Partial<CustomField>) => {
const response = await api.post('/method.call/livechat:saveCustomField', {
message: JSON.stringify({
method: 'livechat:saveCustomField',
params: [
null,
{
field: overwrides.field,
label: overwrides.label || overwrides.field,
visibility: 'visible',
scope: 'visitor',
searchable: false,
regexp: '',
type: 'input',
required: false,
defaultValue: '',
options: '',
public: false,
...overwrides,
},
],
id: 'id',
msg: 'method',
}),
export const createCustomField = async (api: BaseTest['api'], overwrites: Partial<CustomField>) => {
const response = await api.post('/livechat/custom-fields.save', {
customFieldId: null,
customFieldData: {
field: overwrites.field,
label: overwrites.label || overwrites.field,
visibility: 'visible',
scope: 'visitor',
searchable: false,
regexp: '',
type: 'input',
required: false,
defaultValue: '',
options: '',
public: false,
...overwrites,
},
});

if (!response.ok()) {
throw new Error(`Failed to create custom field [http status: ${response.status()}]`);
}

const customField = await parseMeteorResponse<CustomField & { _id: string }>(response);
const { customField } = await response.json();

return {
response,
Expand Down
37 changes: 15 additions & 22 deletions apps/meteor/tests/end-to-end/api/livechat/00-rooms.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2838,30 +2838,23 @@ describe('LIVECHAT - rooms', () => {
});
it('should throw an error if a valid custom field fails the check', async () => {
await request
.post(methodCall('livechat:saveCustomField'))
.post(api('livechat/custom-fields.save'))
.set(credentials)
.send({
message: JSON.stringify({
method: 'livechat:saveCustomField',
params: [
null,
{
field: 'intfield',
label: 'intfield',
scope: 'room',
visibility: 'visible',
regexp: '\\d+',
searchable: true,
type: 'input',
required: false,
defaultValue: '0',
options: '',
public: false,
},
],
id: 'id',
msg: 'method',
}),
customFieldId: null,
customFieldData: {
field: 'intfield',
label: 'intfield',
scope: 'room',
visibility: 'visible',
regexp: '\\d+',
searchable: true,
type: 'input',
required: false,
defaultValue: '0',
options: '',
public: false,
},
})
.expect(200);

Expand Down
Loading
Loading