discourse/app/assets/javascripts/admin/addon/models/web-hook.js
Daniel Waterworth cead0cf684
DEV: Remove Category.findById from admin web hooks (#26605)
Include categories when fetching admin/web_hooks and make
'extras' more useful. 'extras' is the mechanism we use to provide
context for rest objects.

However, previously:

 * When you fetched many objects, extras was only set on the ResultSet,
   not on each object,

 * If you need derived data from extras, there wasn't a sensible place to
   put this code. Now, you can create an 'ExtrasClass' static field on
   your rest model and this class will be used for your extras data,
2024-04-11 16:11:00 -05:00

134 lines
3.6 KiB
JavaScript

import { tracked } from "@glimmer/tracking";
import { computed } from "@ember/object";
import { isEmpty } from "@ember/utils";
import { observes } from "@ember-decorators/object";
import Group from "discourse/models/group";
import RestModel from "discourse/models/rest";
import Site from "discourse/models/site";
import discourseComputed from "discourse-common/utils/decorators";
class WebHookExtras {
@tracked categories;
constructor(args) {
this.categories = args.categories;
this.content_types = args.content_types;
this.default_event_types = args.default_event_types;
this.delivery_statuses = args.delivery_statuses;
this.grouped_event_types = args.grouped_event_types;
}
get categoriesById() {
if (this.categories) {
return new Map(this.categories.map((c) => [c.id, c]));
}
}
findCategoryById(id) {
return this.categoriesById?.get(id);
}
}
export default class WebHook extends RestModel {
static ExtrasClass = WebHookExtras;
content_type = 1; // json
last_delivery_status = 1; // inactive
wildcard_web_hook = false;
verify_certificate = true;
active = false;
web_hook_event_types = null;
groupsFilterInName = null;
@computed("wildcard_web_hook")
get wildcard() {
return this.wildcard_web_hook ? "wildcard" : "individual";
}
set wildcard(value) {
this.set("wildcard_web_hook", value === "wildcard");
}
@computed("category_ids")
get categories() {
return (this.category_ids || []).map((id) =>
this.extras.findCategoryById(id)
);
}
set categories(value) {
this.set(
"category_ids",
value.map((c) => c.id)
);
}
@observes("group_ids")
updateGroupsFilter() {
const groupIds = this.group_ids;
this.set(
"groupsFilterInName",
Site.currentProp("groups").reduce((groupNames, g) => {
if (groupIds.includes(g.id)) {
groupNames.push(g.name);
}
return groupNames;
}, [])
);
}
groupFinder(term) {
return Group.findAll({ term, ignore_automatic: false });
}
@discourseComputed("wildcard_web_hook", "web_hook_event_types.[]")
description(isWildcardWebHook, types) {
let desc = "";
types.forEach((type) => {
const name = `${type.name.toLowerCase()}_event`;
desc += desc !== "" ? `, ${name}` : name;
});
return isWildcardWebHook ? "*" : desc;
}
createProperties() {
const types = this.web_hook_event_types;
const categoryIds = this.categories.map((c) => c.id);
const tagNames = this.tag_names;
// Hack as {{group-selector}} accepts a comma-separated string as data source, but
// we use an array to populate the datasource above.
const groupsFilter = this.groupsFilterInName;
const groupNames =
typeof groupsFilter === "string" ? groupsFilter.split(",") : groupsFilter;
return {
payload_url: this.payload_url,
content_type: this.content_type,
secret: this.secret,
wildcard_web_hook: this.wildcard_web_hook,
verify_certificate: this.verify_certificate,
active: this.active,
web_hook_event_type_ids: isEmpty(types)
? [null]
: types.map((type) => type.id),
category_ids: isEmpty(categoryIds) ? [null] : categoryIds,
tag_names: isEmpty(tagNames) ? [null] : tagNames,
group_ids:
isEmpty(groupNames) || isEmpty(groupNames[0])
? [null]
: Site.currentProp("groups").reduce((groupIds, g) => {
if (groupNames.includes(g.name)) {
groupIds.push(g.id);
}
return groupIds;
}, []),
};
}
updateProperties() {
return this.createProperties();
}
}