forked from Ivasoft/mattermost-mobile
255 lines
6.7 KiB
TypeScript
255 lines
6.7 KiB
TypeScript
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
import {AppBindingLocations, AppCallResponseTypes, AppFieldTypes} from '@constants/apps';
|
|
|
|
import {generateId} from './general';
|
|
|
|
export function cleanBinding(binding: AppBinding, topLocation: string): AppBinding {
|
|
return cleanBindingRec(binding, topLocation, 0);
|
|
}
|
|
|
|
function cleanBindingRec(binding: AppBinding, topLocation: string, depth: number): AppBinding {
|
|
if (!binding) {
|
|
return binding;
|
|
}
|
|
|
|
const toRemove: number[] = [];
|
|
const usedLabels: {[label: string]: boolean} = {};
|
|
binding.bindings?.forEach((b, i) => {
|
|
// Inheritance and defaults
|
|
if (!b.app_id) {
|
|
b.app_id = binding.app_id;
|
|
}
|
|
|
|
if (!b.label) {
|
|
b.label = b.location || '';
|
|
}
|
|
|
|
if (!b.location) {
|
|
b.location = generateId();
|
|
}
|
|
|
|
b.location = binding.location + '/' + b.location;
|
|
|
|
// Validation
|
|
if (!b.app_id) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
|
|
// No empty labels nor "whitespace" labels
|
|
if (!b.label.trim()) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
|
|
switch (topLocation) {
|
|
case AppBindingLocations.COMMAND: {
|
|
if (b.label.match(/ |\t/)) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
|
|
if (usedLabels[b.label]) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
break;
|
|
}
|
|
case AppBindingLocations.IN_POST: {
|
|
if (usedLabels[b.label]) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Must have only subbindings, a form or a submit call.
|
|
const hasBindings = Boolean(b.bindings?.length);
|
|
const hasForm = Boolean(b.form);
|
|
const hasSubmit = Boolean(b.submit);
|
|
if ((!hasBindings && !hasForm && !hasSubmit) ||
|
|
(hasBindings && hasForm) ||
|
|
(hasBindings && hasSubmit) ||
|
|
(hasForm && hasSubmit)) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
|
|
if (hasBindings) {
|
|
cleanBindingRec(b, topLocation, depth + 1);
|
|
|
|
// Remove invalid branches
|
|
if (!b.bindings?.length) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
} else if (hasForm) {
|
|
if (!b.form?.submit && !b.form?.source) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
|
|
cleanForm(b.form);
|
|
}
|
|
|
|
usedLabels[b.label] = true;
|
|
});
|
|
|
|
toRemove.forEach((i) => {
|
|
binding.bindings?.splice(i, 1);
|
|
});
|
|
|
|
return binding;
|
|
}
|
|
|
|
export function validateBindings(bindings: AppBinding[] = []): AppBinding[] {
|
|
const channelHeaderBindings = bindings?.filter((b) => b.location === AppBindingLocations.CHANNEL_HEADER_ICON);
|
|
const postMenuBindings = bindings?.filter((b) => b.location === AppBindingLocations.POST_MENU_ITEM);
|
|
const commandBindings = bindings?.filter((b) => b.location === AppBindingLocations.COMMAND);
|
|
|
|
channelHeaderBindings.forEach((b) => cleanBinding(b, AppBindingLocations.CHANNEL_HEADER_ICON));
|
|
postMenuBindings.forEach((b) => cleanBinding(b, AppBindingLocations.POST_MENU_ITEM));
|
|
commandBindings.forEach((b) => cleanBinding(b, AppBindingLocations.COMMAND));
|
|
|
|
const hasBindings = (b: AppBinding) => b.bindings?.length;
|
|
return postMenuBindings.filter(hasBindings).concat(channelHeaderBindings.filter(hasBindings), commandBindings.filter(hasBindings));
|
|
}
|
|
|
|
export function cleanForm(form?: AppForm): void {
|
|
if (!form) {
|
|
return;
|
|
}
|
|
|
|
const toRemove: number[] = [];
|
|
const usedLabels: {[label: string]: boolean} = {};
|
|
form.fields?.forEach((field, i) => {
|
|
if (!field.name) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
|
|
if (field.name.match(/ |\t/)) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
|
|
let label = field.label;
|
|
if (!label) {
|
|
label = field.name;
|
|
}
|
|
|
|
if (label.match(/ |\t/)) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
|
|
if (usedLabels[label]) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
|
|
switch (field.type) {
|
|
case AppFieldTypes.STATIC_SELECT:
|
|
cleanStaticSelect(field);
|
|
if (!field.options?.length) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
break;
|
|
case AppFieldTypes.DYNAMIC_SELECT:
|
|
if (!field.lookup) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
}
|
|
|
|
usedLabels[label] = true;
|
|
});
|
|
|
|
toRemove.forEach((i) => {
|
|
form.fields!.splice(i, 1);
|
|
});
|
|
}
|
|
|
|
function cleanStaticSelect(field: AppField): void {
|
|
const toRemove: number[] = [];
|
|
const usedLabels: {[label: string]: boolean} = {};
|
|
const usedValues: {[label: string]: boolean} = {};
|
|
field.options?.forEach((option, i) => {
|
|
let label = option.label;
|
|
if (!label) {
|
|
label = option.value;
|
|
}
|
|
|
|
if (!label) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
|
|
if (usedLabels[label]) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
|
|
if (usedValues[option.value]) {
|
|
toRemove.unshift(i);
|
|
return;
|
|
}
|
|
|
|
usedLabels[label] = true;
|
|
usedValues[option.value] = true;
|
|
});
|
|
|
|
toRemove.forEach((i) => {
|
|
field.options?.splice(i, 1);
|
|
});
|
|
}
|
|
|
|
export function createCallContext(
|
|
appID: string,
|
|
location?: string,
|
|
channelID?: string,
|
|
teamID?: string,
|
|
postID?: string,
|
|
rootID?: string,
|
|
): AppContext {
|
|
return {
|
|
app_id: appID,
|
|
location,
|
|
channel_id: channelID,
|
|
team_id: teamID,
|
|
post_id: postID,
|
|
root_id: rootID,
|
|
};
|
|
}
|
|
|
|
export function createCallRequest(
|
|
call: AppCall,
|
|
context: AppContext,
|
|
defaultExpand: AppExpand = {},
|
|
values?: AppCallValues,
|
|
rawCommand?: string,
|
|
): AppCallRequest {
|
|
return {
|
|
...call,
|
|
context,
|
|
values,
|
|
expand: {
|
|
...defaultExpand,
|
|
...call.expand,
|
|
},
|
|
raw_command: rawCommand,
|
|
};
|
|
}
|
|
|
|
export const makeCallErrorResponse = (errMessage: string): AppCallResponse<any> => {
|
|
return {
|
|
type: AppCallResponseTypes.ERROR,
|
|
text: errMessage,
|
|
};
|
|
};
|
|
|
|
export const filterEmptyOptions = (option: AppSelectOption): boolean => Boolean(option.value && !option.value.match(/^[ \t]+$/));
|