2020-02-21 20:46:35 +01:00
|
|
|
const settings_config = require("./settings_config");
|
2019-11-02 00:06:25 +01:00
|
|
|
const render_settings_admin_auth_methods_list = require('../templates/settings/admin_auth_methods_list.hbs');
|
|
|
|
const render_settings_admin_realm_domains_list = require("../templates/settings/admin_realm_domains_list.hbs");
|
|
|
|
const render_settings_admin_realm_dropdown_stream_list = require("../templates/settings/admin_realm_dropdown_stream_list.hbs");
|
|
|
|
const render_settings_organization_settings_tip = require("../templates/settings/organization_settings_tip.hbs");
|
2020-03-31 15:21:27 +02:00
|
|
|
const pygments_data = require("../generated/pygments_data.json");
|
2019-07-09 21:24:00 +02:00
|
|
|
|
2019-11-02 00:06:25 +01:00
|
|
|
const meta = {
|
2017-04-08 18:13:39 +02:00
|
|
|
loaded: false,
|
|
|
|
};
|
|
|
|
|
2020-03-31 15:21:27 +02:00
|
|
|
exports.default_code_language_widget = (function (element_id) {
|
|
|
|
|
|
|
|
const render_language_list = require("../templates/settings/admin_realm_dropdown_code_languages_list.hbs");
|
|
|
|
const language_list = Object.keys(pygments_data.langs).map(x => {
|
|
|
|
return {
|
|
|
|
name: x,
|
|
|
|
priority: pygments_data.langs[x],
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
const setup = () => {
|
|
|
|
// populate the dropdown
|
|
|
|
const dropdown_list_body = $(`#${element_id} .dropdown-list-body`).expectOne();
|
|
|
|
const search_input = $(`#${element_id} .dropdown-search > input[type=text]`);
|
|
|
|
list_render.create(dropdown_list_body, language_list, {
|
|
|
|
name: "admin-realm-default-code-language-dropdown-list",
|
|
|
|
modifier: function (item) {
|
|
|
|
return render_language_list({ language: item });
|
|
|
|
},
|
|
|
|
filter: {
|
|
|
|
element: search_input,
|
|
|
|
predicate: function (item, value) {
|
|
|
|
return item.name.toLowerCase().includes(value);
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}).init();
|
|
|
|
$(`#${element_id} .dropdown-search`).click(function (e) {
|
|
|
|
e.stopPropagation();
|
|
|
|
});
|
|
|
|
|
|
|
|
$(`#${element_id} .dropdown-toggle`).click(function () {
|
|
|
|
search_input.val("").trigger("input");
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const render = (name) => {
|
|
|
|
$(`#${element_id} #id_realm_default_code_block_language`).data("language", name);
|
|
|
|
|
|
|
|
const elem = $(`#${element_id} #realm_default_code_block_language_name`);
|
|
|
|
|
|
|
|
if (!name) {
|
|
|
|
elem.text(i18n.t("No language set"));
|
|
|
|
elem.addClass("text-warning");
|
|
|
|
elem.closest('.input-group').find('.default_code_block_language_unset').hide();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Happy path
|
|
|
|
elem.text(name);
|
|
|
|
elem.removeClass('text-warning');
|
|
|
|
elem.closest('.input-group').find('.default_code_block_language_unset').show();
|
|
|
|
};
|
|
|
|
|
|
|
|
const update = (lang, save_discard_widget_status_handler) => {
|
|
|
|
render(lang);
|
|
|
|
save_discard_widget_status_handler($('#org-other-settings'));
|
|
|
|
};
|
|
|
|
|
|
|
|
const register_event_handlers = (save_discard_widget_status_handler) => {
|
|
|
|
$(`#${element_id} .dropdown-list-body`).on("click keypress", ".lang_name", function (e) {
|
|
|
|
const setting_elem = $(this).closest(".realm_default_code_block_language_setting");
|
|
|
|
if (e.type === "keypress") {
|
|
|
|
if (e.which === 13) {
|
|
|
|
setting_elem.find(".dropdown-menu").dropdown("toggle");
|
|
|
|
} else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const lang = $(this).attr('data-language');
|
|
|
|
update(lang, save_discard_widget_status_handler);
|
|
|
|
});
|
|
|
|
$(`#${element_id} .default_code_block_language_unset`).click(function () {
|
|
|
|
update(null, save_discard_widget_status_handler);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const value = () => {
|
|
|
|
let val = $(`#${element_id} #id_realm_default_code_block_language`).data('language');
|
|
|
|
if (val === null) {
|
|
|
|
val = '';
|
|
|
|
}
|
|
|
|
return val;
|
|
|
|
};
|
|
|
|
|
|
|
|
return {
|
|
|
|
setup,
|
|
|
|
render,
|
|
|
|
register_event_handlers,
|
|
|
|
value,
|
|
|
|
};
|
|
|
|
}('realm_default_code_block_language_widget'));
|
|
|
|
|
2017-04-17 16:51:27 +02:00
|
|
|
exports.reset = function () {
|
|
|
|
meta.loaded = false;
|
|
|
|
};
|
|
|
|
|
2018-08-03 22:52:21 +02:00
|
|
|
|
2018-12-08 17:37:57 +01:00
|
|
|
exports.maybe_disable_widgets = function () {
|
|
|
|
if (page_params.is_admin) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$(".organization-box [data-name='organization-profile']")
|
|
|
|
.find("input, textarea, button, select").attr("disabled", true);
|
|
|
|
|
|
|
|
$(".organization-box [data-name='organization-settings']")
|
|
|
|
.find("input, textarea, button, select").attr("disabled", true);
|
|
|
|
|
|
|
|
$(".organization-box [data-name='organization-settings']")
|
|
|
|
.find(".control-label-disabled").addClass('enabled');
|
|
|
|
|
|
|
|
$(".organization-box [data-name='organization-permissions']")
|
|
|
|
.find("input, textarea, button, select").attr("disabled", true);
|
|
|
|
|
|
|
|
$(".organization-box [data-name='organization-permissions']")
|
|
|
|
.find(".control-label-disabled").addClass('enabled');
|
|
|
|
|
|
|
|
$(".organization-box [data-name='auth-methods']")
|
|
|
|
.find("input, button, select, checked").attr("disabled", true);
|
|
|
|
};
|
|
|
|
|
2020-01-23 07:01:34 +01:00
|
|
|
exports.get_sorted_options_list = function (option_values_object) {
|
2020-02-09 04:15:38 +01:00
|
|
|
const options_list = Object.keys(option_values_object).map(key => ({
|
|
|
|
...option_values_object[key],
|
|
|
|
key: key,
|
|
|
|
}));
|
2020-01-23 07:01:34 +01:00
|
|
|
let comparator = (x, y) => x.order - y.order;
|
|
|
|
if (!options_list[0].order) {
|
|
|
|
comparator = (x, y) => {
|
|
|
|
const key_x = x.key.toUpperCase();
|
|
|
|
const key_y = y.key.toUpperCase();
|
|
|
|
if (key_x < key_y) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (key_x > key_y) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
options_list.sort(comparator);
|
|
|
|
return options_list;
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.get_organization_settings_options = () => {
|
|
|
|
const options = {};
|
|
|
|
options.create_stream_policy_values = exports.get_sorted_options_list(
|
2020-02-21 20:58:17 +01:00
|
|
|
settings_config.create_stream_policy_values);
|
2020-01-25 10:23:40 +01:00
|
|
|
options.invite_to_stream_policy_values = exports.get_sorted_options_list(
|
2020-02-21 20:58:17 +01:00
|
|
|
settings_config.invite_to_stream_policy_values);
|
2020-01-26 16:55:09 +01:00
|
|
|
options.user_group_edit_policy_values = exports.get_sorted_options_list(
|
2020-02-21 21:40:29 +01:00
|
|
|
settings_config.user_group_edit_policy_values);
|
2020-01-28 07:51:44 +01:00
|
|
|
options.private_message_policy_values = exports.get_sorted_options_list(
|
2020-02-21 21:42:27 +01:00
|
|
|
settings_config.private_message_policy_values);
|
2020-01-23 07:01:34 +01:00
|
|
|
return options;
|
|
|
|
};
|
|
|
|
|
2018-04-28 15:35:14 +02:00
|
|
|
exports.get_realm_time_limits_in_minutes = function (property) {
|
2019-11-02 00:06:25 +01:00
|
|
|
let val = (page_params[property] / 60).toFixed(1);
|
2018-04-28 15:35:14 +02:00
|
|
|
if (parseFloat(val, 10) === parseInt(val, 10)) {
|
|
|
|
val = parseInt(val, 10);
|
|
|
|
}
|
|
|
|
return val.toString();
|
|
|
|
};
|
|
|
|
|
2018-04-01 08:56:18 +02:00
|
|
|
function get_property_value(property_name) {
|
2018-03-29 12:52:57 +02:00
|
|
|
if (property_name === 'realm_message_content_edit_limit_minutes') {
|
2018-04-28 15:35:14 +02:00
|
|
|
return exports.get_realm_time_limits_in_minutes('realm_message_content_edit_limit_seconds');
|
2018-06-16 13:27:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (property_name === 'realm_message_content_delete_limit_minutes') {
|
2018-04-28 15:35:14 +02:00
|
|
|
return exports.get_realm_time_limits_in_minutes('realm_message_content_delete_limit_seconds');
|
2018-06-16 13:27:56 +02:00
|
|
|
}
|
|
|
|
|
2019-05-07 19:06:05 +02:00
|
|
|
if (property_name === 'realm_waiting_period_setting') {
|
|
|
|
if (page_params.realm_waiting_period_threshold === 0) {
|
|
|
|
return "none";
|
|
|
|
}
|
|
|
|
if (page_params.realm_waiting_period_threshold === 3) {
|
|
|
|
return "three_days";
|
|
|
|
}
|
|
|
|
return "custom_days";
|
|
|
|
}
|
|
|
|
|
2018-06-16 13:27:56 +02:00
|
|
|
if (property_name === 'realm_add_emoji_by_admins_only') {
|
2018-03-29 12:52:57 +02:00
|
|
|
if (page_params.realm_add_emoji_by_admins_only) {
|
|
|
|
return "by_admins_only";
|
|
|
|
}
|
|
|
|
return "by_anyone";
|
2018-06-16 13:27:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (property_name === 'realm_msg_edit_limit_setting') {
|
2018-04-02 11:18:40 +02:00
|
|
|
if (!page_params.realm_allow_message_editing) {
|
|
|
|
return "never";
|
|
|
|
}
|
2020-02-09 05:23:25 +01:00
|
|
|
for (const [value, elem] of settings_config.msg_edit_limit_dropdown_values) {
|
|
|
|
if (elem.seconds === page_params.realm_message_content_edit_limit_seconds) {
|
|
|
|
return value;
|
|
|
|
}
|
2018-04-02 11:18:40 +02:00
|
|
|
}
|
2020-02-09 05:23:25 +01:00
|
|
|
return "custom_limit";
|
2018-06-16 13:27:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (property_name === 'realm_msg_delete_limit_setting') {
|
2017-11-26 09:12:10 +01:00
|
|
|
if (!page_params.realm_allow_message_deleting) {
|
|
|
|
return "never";
|
|
|
|
}
|
2020-02-09 05:23:25 +01:00
|
|
|
for (const [value, elem] of settings_config.msg_delete_limit_dropdown_values) {
|
|
|
|
if (elem.seconds === page_params.realm_message_content_delete_limit_seconds) {
|
|
|
|
return value;
|
|
|
|
}
|
2017-11-26 09:12:10 +01:00
|
|
|
}
|
2020-02-09 05:23:25 +01:00
|
|
|
return "custom_limit";
|
2018-06-16 13:27:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (property_name === 'realm_org_join_restrictions') {
|
2018-07-27 23:26:29 +02:00
|
|
|
if (page_params.realm_emails_restricted_to_domains) {
|
2018-04-05 00:54:31 +02:00
|
|
|
return "only_selected_domain";
|
|
|
|
}
|
|
|
|
if (page_params.realm_disallow_disposable_email_addresses) {
|
|
|
|
return "no_disposable_email";
|
|
|
|
}
|
|
|
|
return "no_restriction";
|
2018-06-16 13:27:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (property_name === 'realm_user_invite_restriction') {
|
2018-04-30 18:26:59 +02:00
|
|
|
if (!page_params.realm_invite_required) {
|
|
|
|
return "no_invite_required";
|
|
|
|
}
|
|
|
|
if (page_params.realm_invite_by_admins_only) {
|
|
|
|
return "by_admins_only";
|
|
|
|
}
|
|
|
|
return "by_anyone";
|
2018-03-29 12:52:57 +02:00
|
|
|
}
|
2018-06-16 13:27:56 +02:00
|
|
|
|
2019-07-30 09:52:06 +02:00
|
|
|
if (property_name === 'realm_default_twenty_four_hour_time') {
|
|
|
|
return JSON.stringify(page_params[property_name]);
|
|
|
|
}
|
|
|
|
|
2020-01-13 12:32:38 +01:00
|
|
|
if (property_name === 'realm_notifications_stream') {
|
|
|
|
return page_params.realm_notifications_stream_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (property_name === 'realm_signup_notifications_stream') {
|
|
|
|
return page_params.realm_signup_notifications_stream_id;
|
|
|
|
}
|
|
|
|
|
2018-04-19 21:45:47 +02:00
|
|
|
return page_params[property_name];
|
2018-03-29 12:52:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
exports.extract_property_name = function (elem) {
|
|
|
|
return elem.attr('id').split('-').join('_').replace("id_", "");
|
|
|
|
};
|
|
|
|
|
2019-05-05 22:37:09 +02:00
|
|
|
function get_subsection_property_elements(element) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const subsection = $(element).closest('.org-subsection-parent');
|
2020-02-16 10:31:49 +01:00
|
|
|
return Array.from(subsection.find('.prop-element'));
|
2019-05-05 22:37:09 +02:00
|
|
|
}
|
|
|
|
|
2020-02-03 12:40:36 +01:00
|
|
|
const simple_dropdown_properties = ['realm_create_stream_policy',
|
|
|
|
'realm_invite_to_stream_policy',
|
|
|
|
'realm_user_group_edit_policy',
|
|
|
|
'realm_private_message_policy',
|
|
|
|
'realm_add_emoji_by_admins_only',
|
|
|
|
'realm_user_invite_restriction'];
|
|
|
|
|
|
|
|
function set_property_dropdown_value(property_name) {
|
|
|
|
$('#id_' + property_name).val(get_property_value(property_name));
|
|
|
|
}
|
|
|
|
|
2020-02-03 13:03:53 +01:00
|
|
|
function change_element_block_display_property(elem_id, show_element) {
|
|
|
|
const elem = $("#" + elem_id);
|
|
|
|
if (show_element) {
|
|
|
|
elem.parent().show();
|
|
|
|
} else {
|
|
|
|
elem.parent().hide();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-07 19:06:05 +02:00
|
|
|
function set_realm_waiting_period_dropdown() {
|
2019-11-02 00:06:25 +01:00
|
|
|
const value = get_property_value("realm_waiting_period_setting");
|
2019-05-07 19:06:05 +02:00
|
|
|
$("#id_realm_waiting_period_setting").val(value);
|
2020-02-03 13:03:53 +01:00
|
|
|
change_element_block_display_property('id_realm_waiting_period_threshold',
|
|
|
|
value === "custom_days");
|
2018-04-01 08:49:10 +02:00
|
|
|
}
|
2018-01-05 15:34:10 +01:00
|
|
|
|
2018-04-23 14:51:30 +02:00
|
|
|
function set_video_chat_provider_dropdown() {
|
2019-11-02 00:06:25 +01:00
|
|
|
const chat_provider_id = page_params.realm_video_chat_provider;
|
|
|
|
const available_providers = page_params.realm_available_video_chat_providers;
|
2019-05-09 09:54:38 +02:00
|
|
|
|
|
|
|
$("#id_realm_video_chat_provider").val(chat_provider_id);
|
|
|
|
if (chat_provider_id === available_providers.google_hangouts.id) {
|
2018-04-23 14:51:30 +02:00
|
|
|
$("#google_hangouts_domain").show();
|
2018-12-28 20:45:54 +01:00
|
|
|
$(".zoom_credentials").hide();
|
2018-04-23 14:51:30 +02:00
|
|
|
$("#id_realm_google_hangouts_domain").val(page_params.realm_google_hangouts_domain);
|
2019-05-09 09:54:38 +02:00
|
|
|
} else if (chat_provider_id === available_providers.zoom.id) {
|
2018-12-28 20:45:54 +01:00
|
|
|
$("#google_hangouts_domain").hide();
|
|
|
|
$(".zoom_credentials").show();
|
|
|
|
$("#id_realm_zoom_user_id").val(page_params.realm_zoom_user_id);
|
|
|
|
$("#id_realm_zoom_api_key").val(page_params.realm_zoom_api_key);
|
|
|
|
$("#id_realm_zoom_api_secret").val(page_params.realm_zoom_api_secret);
|
2018-04-23 14:51:30 +02:00
|
|
|
} else {
|
|
|
|
$("#google_hangouts_domain").hide();
|
2018-12-28 20:45:54 +01:00
|
|
|
$(".zoom_credentials").hide();
|
2018-04-23 14:51:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-02 11:18:40 +02:00
|
|
|
function set_msg_edit_limit_dropdown() {
|
2019-11-02 00:06:25 +01:00
|
|
|
const value = get_property_value("realm_msg_edit_limit_setting");
|
2018-04-02 11:18:40 +02:00
|
|
|
$("#id_realm_msg_edit_limit_setting").val(value);
|
2020-02-03 13:03:53 +01:00
|
|
|
change_element_block_display_property('id_realm_message_content_edit_limit_minutes',
|
|
|
|
value === "custom_limit");
|
2018-08-18 12:31:21 +02:00
|
|
|
settings_ui.disable_sub_setting_onchange(value !== "never",
|
|
|
|
"id_realm_allow_community_topic_editing", true);
|
2018-04-02 11:18:40 +02:00
|
|
|
}
|
|
|
|
|
2017-11-26 09:12:10 +01:00
|
|
|
function set_msg_delete_limit_dropdown() {
|
2019-11-02 00:06:25 +01:00
|
|
|
const value = get_property_value("realm_msg_delete_limit_setting");
|
2017-11-26 09:12:10 +01:00
|
|
|
$("#id_realm_msg_delete_limit_setting").val(value);
|
2020-02-03 13:03:53 +01:00
|
|
|
change_element_block_display_property('id_realm_message_content_delete_limit_minutes',
|
|
|
|
value === "custom_limit");
|
2017-11-26 09:12:10 +01:00
|
|
|
}
|
|
|
|
|
2018-04-05 00:54:31 +02:00
|
|
|
function set_org_join_restrictions_dropdown() {
|
2019-11-02 00:06:25 +01:00
|
|
|
const value = get_property_value("realm_org_join_restrictions");
|
2018-04-05 00:54:31 +02:00
|
|
|
$("#id_realm_org_join_restrictions").val(value);
|
2020-02-03 13:03:53 +01:00
|
|
|
change_element_block_display_property('allowed_domains_label',
|
|
|
|
value === 'only_selected_domain');
|
2018-04-05 00:54:31 +02:00
|
|
|
}
|
|
|
|
|
2019-01-14 14:04:08 +01:00
|
|
|
function set_message_content_in_email_notifications_visiblity() {
|
2020-02-03 13:03:53 +01:00
|
|
|
change_element_block_display_property(
|
|
|
|
'message_content_in_email_notifications_label',
|
|
|
|
page_params.realm_message_content_allowed_in_email_notifications);
|
2019-01-14 14:04:08 +01:00
|
|
|
}
|
|
|
|
|
2019-05-08 08:11:30 +02:00
|
|
|
function set_digest_emails_weekday_visibility() {
|
2020-02-03 13:03:53 +01:00
|
|
|
change_element_block_display_property('id_realm_digest_weekday',
|
|
|
|
page_params.realm_digest_emails_enabled);
|
2019-05-08 08:11:30 +02:00
|
|
|
}
|
|
|
|
|
2017-04-08 18:13:39 +02:00
|
|
|
exports.populate_realm_domains = function (realm_domains) {
|
|
|
|
if (!meta.loaded) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
js: Convert _.map(a, …) to a.map(…).
And convert the corresponding function expressions to arrow style
while we’re here.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import K from "ast-types/gen/kinds";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
recast.visit(ast, {
visitCallExpression(path) {
const { callee, arguments: args } = path.node;
if (
n.MemberExpression.check(callee) &&
!callee.computed &&
n.Identifier.check(callee.object) &&
callee.object.name === "_" &&
n.Identifier.check(callee.property) &&
callee.property.name === "map" &&
args.length === 2 &&
checkExpression(args[0]) &&
checkExpression(args[1])
) {
const [arr, fn] = args;
path.replace(
b.callExpression(b.memberExpression(arr, b.identifier("map")), [
n.FunctionExpression.check(fn) ||
n.ArrowFunctionExpression.check(fn)
? b.arrowFunctionExpression(
fn.params,
n.BlockStatement.check(fn.body) &&
fn.body.body.length === 1 &&
n.ReturnStatement.check(fn.body.body[0])
? fn.body.body[0].argument || b.identifier("undefined")
: fn.body
)
: fn,
])
);
changed = true;
}
this.traverse(path);
},
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-08 02:43:49 +01:00
|
|
|
const domains_list = realm_domains.map(
|
|
|
|
realm_domain => realm_domain.allow_subdomains ? "*." + realm_domain.domain : realm_domain.domain
|
|
|
|
);
|
2019-11-02 00:06:25 +01:00
|
|
|
let domains = domains_list.join(', ');
|
2017-04-08 18:13:39 +02:00
|
|
|
if (domains.length === 0) {
|
|
|
|
domains = i18n.t("None");
|
|
|
|
}
|
2018-04-05 00:54:31 +02:00
|
|
|
$("#allowed_domains_label").text(i18n.t("Allowed domains: __domains__", {domains: domains}));
|
2017-04-08 18:13:39 +02:00
|
|
|
|
2019-11-02 00:06:25 +01:00
|
|
|
const realm_domains_table_body = $("#realm_domains_table tbody").expectOne();
|
2017-04-08 18:13:39 +02:00
|
|
|
realm_domains_table_body.find("tr").remove();
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
|
|
|
|
for (const realm_domain of realm_domains) {
|
2019-07-09 21:24:00 +02:00
|
|
|
realm_domains_table_body.append(
|
|
|
|
render_settings_admin_realm_domains_list({
|
|
|
|
realm_domain: realm_domain,
|
|
|
|
})
|
|
|
|
);
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
}
|
2017-04-08 18:13:39 +02:00
|
|
|
};
|
2019-04-07 18:56:32 +02:00
|
|
|
function sort_object_by_key(obj) {
|
2020-02-09 04:45:48 +01:00
|
|
|
const keys = Object.keys(obj).sort();
|
2019-11-02 00:06:25 +01:00
|
|
|
const new_obj = {};
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
|
|
|
|
for (const key of keys) {
|
2019-04-07 18:56:32 +02:00
|
|
|
new_obj[key] = obj[key];
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
}
|
|
|
|
|
2019-04-07 18:56:32 +02:00
|
|
|
return new_obj;
|
|
|
|
}
|
2017-04-08 18:13:39 +02:00
|
|
|
exports.populate_auth_methods = function (auth_methods) {
|
|
|
|
if (!meta.loaded) {
|
|
|
|
return;
|
|
|
|
}
|
2019-11-02 00:06:25 +01:00
|
|
|
const auth_methods_table = $("#id_realm_authentication_methods").expectOne();
|
2019-04-07 18:56:32 +02:00
|
|
|
auth_methods = sort_object_by_key(auth_methods);
|
2019-11-02 00:06:25 +01:00
|
|
|
let rendered_auth_method_rows = "";
|
2020-02-06 04:27:31 +01:00
|
|
|
for (const [auth_method, value] of Object.entries(auth_methods)) {
|
2019-07-09 21:24:00 +02:00
|
|
|
rendered_auth_method_rows += render_settings_admin_auth_methods_list({
|
2019-04-06 08:43:34 +02:00
|
|
|
method: auth_method,
|
|
|
|
enabled: value,
|
|
|
|
is_admin: page_params.is_admin,
|
|
|
|
});
|
2020-02-06 04:27:31 +01:00
|
|
|
}
|
2019-04-06 08:43:34 +02:00
|
|
|
auth_methods_table.html(rendered_auth_method_rows);
|
2017-04-08 18:13:39 +02:00
|
|
|
};
|
|
|
|
|
2019-04-07 18:32:42 +02:00
|
|
|
function insert_tip_box() {
|
|
|
|
if (page_params.is_admin) {
|
|
|
|
return;
|
|
|
|
}
|
2019-11-02 00:06:25 +01:00
|
|
|
const tip_box = render_settings_organization_settings_tip({is_admin: page_params.is_admin});
|
2019-04-07 18:32:42 +02:00
|
|
|
$(".organization-box").find(".settings-section:not(.can-edit)")
|
|
|
|
.not("#emoji-settings")
|
|
|
|
.not("#user-groups-admin")
|
|
|
|
.prepend(tip_box);
|
|
|
|
}
|
|
|
|
|
2020-01-11 21:55:17 +01:00
|
|
|
exports.render_notifications_stream_ui = function (stream_id, notification_type) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const name = stream_data.maybe_get_stream_name(stream_id);
|
2017-06-16 14:39:21 +02:00
|
|
|
|
2020-01-11 21:55:17 +01:00
|
|
|
$(`#id_realm_${notification_type}_stream`).data("stream-id", stream_id);
|
|
|
|
|
|
|
|
const elem = $(`#realm_${notification_type}_stream_name`);
|
|
|
|
|
2017-06-16 14:39:21 +02:00
|
|
|
if (!name) {
|
|
|
|
elem.text(i18n.t("Disabled"));
|
|
|
|
elem.addClass("text-warning");
|
2018-04-03 22:25:18 +02:00
|
|
|
elem.closest('.input-group').find('.notification-disable').hide();
|
2017-06-16 14:39:21 +02:00
|
|
|
return;
|
2017-06-10 07:03:53 +02:00
|
|
|
}
|
2017-06-16 14:39:21 +02:00
|
|
|
|
|
|
|
// Happy path
|
|
|
|
elem.text('#' + name);
|
|
|
|
elem.removeClass('text-warning');
|
2018-04-03 22:25:18 +02:00
|
|
|
elem.closest('.input-group').find('.notification-disable').show();
|
2017-06-10 07:03:53 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
exports.populate_notifications_stream_dropdown = function (stream_list) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const dropdown_list_body = $("#id_realm_notifications_stream .dropdown-list-body").expectOne();
|
|
|
|
const search_input = $("#id_realm_notifications_stream .dropdown-search > input[type=text]");
|
2017-06-10 07:03:53 +02:00
|
|
|
|
2018-06-20 18:32:39 +02:00
|
|
|
list_render.create(dropdown_list_body, stream_list, {
|
2017-10-20 16:55:04 +02:00
|
|
|
name: "admin-realm-notifications-stream-dropdown-list",
|
2017-06-10 07:03:53 +02:00
|
|
|
modifier: function (item) {
|
2019-07-09 21:24:00 +02:00
|
|
|
return render_settings_admin_realm_dropdown_stream_list({ stream: item });
|
2017-06-10 07:03:53 +02:00
|
|
|
},
|
|
|
|
filter: {
|
|
|
|
element: search_input,
|
2019-12-30 17:44:24 +01:00
|
|
|
predicate: function (item, value) {
|
js: Convert a.indexOf(…) !== -1 to a.includes(…).
Babel polyfills this for us for Internet Explorer.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import K from "ast-types/gen/kinds";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
recast.visit(ast, {
visitBinaryExpression(path) {
const { operator, left, right } = path.node;
if (
n.CallExpression.check(left) &&
n.MemberExpression.check(left.callee) &&
!left.callee.computed &&
n.Identifier.check(left.callee.property) &&
left.callee.property.name === "indexOf" &&
left.arguments.length === 1 &&
checkExpression(left.arguments[0]) &&
((["===", "!==", "==", "!=", ">", "<="].includes(operator) &&
n.UnaryExpression.check(right) &&
right.operator == "-" &&
n.Literal.check(right.argument) &&
right.argument.value === 1) ||
([">=", "<"].includes(operator) &&
n.Literal.check(right) &&
right.value === 0))
) {
const test = b.callExpression(
b.memberExpression(left.callee.object, b.identifier("includes")),
[left.arguments[0]]
);
path.replace(
["!==", "!=", ">", ">="].includes(operator)
? test
: b.unaryExpression("!", test)
);
changed = true;
}
this.traverse(path);
},
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-08 04:55:06 +01:00
|
|
|
return item.name.toLowerCase().includes(value);
|
2017-06-10 07:03:53 +02:00
|
|
|
},
|
2017-09-27 19:01:04 +02:00
|
|
|
onupdate: function () {
|
2019-01-09 14:30:35 +01:00
|
|
|
ui.reset_scrollbar(dropdown_list_body);
|
2017-09-27 19:01:04 +02:00
|
|
|
},
|
2017-06-10 07:03:53 +02:00
|
|
|
},
|
|
|
|
}).init();
|
|
|
|
|
|
|
|
$("#id_realm_notifications_stream .dropdown-search").click(function (e) {
|
|
|
|
e.stopPropagation();
|
|
|
|
});
|
|
|
|
|
|
|
|
$("#id_realm_notifications_stream .dropdown-toggle").click(function () {
|
|
|
|
search_input.val("").trigger("input");
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2017-10-20 16:55:04 +02:00
|
|
|
exports.populate_signup_notifications_stream_dropdown = function (stream_list) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const dropdown_list_body = $("#id_realm_signup_notifications_stream .dropdown-list-body").expectOne();
|
|
|
|
const search_input = $("#id_realm_signup_notifications_stream .dropdown-search > input[type=text]");
|
2017-10-20 16:55:04 +02:00
|
|
|
|
2018-06-20 18:32:39 +02:00
|
|
|
list_render.create(dropdown_list_body, stream_list, {
|
2017-10-20 16:55:04 +02:00
|
|
|
name: "admin-realm-signup-notifications-stream-dropdown-list",
|
|
|
|
modifier: function (item) {
|
2019-07-09 21:24:00 +02:00
|
|
|
return render_settings_admin_realm_dropdown_stream_list({ stream: item });
|
2017-10-20 16:55:04 +02:00
|
|
|
},
|
|
|
|
filter: {
|
|
|
|
element: search_input,
|
2019-12-30 17:44:24 +01:00
|
|
|
predicate: function (item, value) {
|
js: Convert a.indexOf(…) !== -1 to a.includes(…).
Babel polyfills this for us for Internet Explorer.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import K from "ast-types/gen/kinds";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
recast.visit(ast, {
visitBinaryExpression(path) {
const { operator, left, right } = path.node;
if (
n.CallExpression.check(left) &&
n.MemberExpression.check(left.callee) &&
!left.callee.computed &&
n.Identifier.check(left.callee.property) &&
left.callee.property.name === "indexOf" &&
left.arguments.length === 1 &&
checkExpression(left.arguments[0]) &&
((["===", "!==", "==", "!=", ">", "<="].includes(operator) &&
n.UnaryExpression.check(right) &&
right.operator == "-" &&
n.Literal.check(right.argument) &&
right.argument.value === 1) ||
([">=", "<"].includes(operator) &&
n.Literal.check(right) &&
right.value === 0))
) {
const test = b.callExpression(
b.memberExpression(left.callee.object, b.identifier("includes")),
[left.arguments[0]]
);
path.replace(
["!==", "!=", ">", ">="].includes(operator)
? test
: b.unaryExpression("!", test)
);
changed = true;
}
this.traverse(path);
},
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-08 04:55:06 +01:00
|
|
|
return item.name.toLowerCase().includes(value);
|
2017-10-20 16:55:04 +02:00
|
|
|
},
|
|
|
|
},
|
|
|
|
}).init();
|
|
|
|
|
|
|
|
$("#id_realm_signup_notifications_stream .dropdown-search").click(function (e) {
|
|
|
|
e.stopPropagation();
|
|
|
|
});
|
|
|
|
|
|
|
|
$("#id_realm_signup_notifications_stream .dropdown-toggle").click(function () {
|
|
|
|
search_input.val("").trigger("input");
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2018-04-01 08:56:18 +02:00
|
|
|
function update_dependent_subsettings(property_name) {
|
js: Convert a.indexOf(…) !== -1 to a.includes(…).
Babel polyfills this for us for Internet Explorer.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import K from "ast-types/gen/kinds";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
recast.visit(ast, {
visitBinaryExpression(path) {
const { operator, left, right } = path.node;
if (
n.CallExpression.check(left) &&
n.MemberExpression.check(left.callee) &&
!left.callee.computed &&
n.Identifier.check(left.callee.property) &&
left.callee.property.name === "indexOf" &&
left.arguments.length === 1 &&
checkExpression(left.arguments[0]) &&
((["===", "!==", "==", "!=", ">", "<="].includes(operator) &&
n.UnaryExpression.check(right) &&
right.operator == "-" &&
n.Literal.check(right.argument) &&
right.argument.value === 1) ||
([">=", "<"].includes(operator) &&
n.Literal.check(right) &&
right.value === 0))
) {
const test = b.callExpression(
b.memberExpression(left.callee.object, b.identifier("includes")),
[left.arguments[0]]
);
path.replace(
["!==", "!=", ">", ">="].includes(operator)
? test
: b.unaryExpression("!", test)
);
changed = true;
}
this.traverse(path);
},
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-08 04:55:06 +01:00
|
|
|
if (simple_dropdown_properties.includes(property_name)) {
|
2020-02-03 12:40:36 +01:00
|
|
|
set_property_dropdown_value(property_name);
|
|
|
|
} else if (property_name === 'realm_waiting_period_threshold') {
|
2019-05-07 19:06:05 +02:00
|
|
|
set_realm_waiting_period_dropdown();
|
2018-05-06 21:43:17 +02:00
|
|
|
} else if (property_name === 'realm_video_chat_provider' ||
|
2018-12-28 20:45:54 +01:00
|
|
|
property_name === 'realm_google_hangouts_domain' ||
|
|
|
|
property_name.startsWith('realm_zoom')) {
|
2018-04-23 14:51:30 +02:00
|
|
|
set_video_chat_provider_dropdown();
|
2018-05-06 21:43:17 +02:00
|
|
|
} else if (property_name === 'realm_msg_edit_limit_setting' ||
|
|
|
|
property_name === 'realm_message_content_edit_limit_minutes') {
|
2018-04-02 11:18:40 +02:00
|
|
|
set_msg_edit_limit_dropdown();
|
2017-11-26 09:12:10 +01:00
|
|
|
} else if (property_name === 'realm_msg_delete_limit_setting' ||
|
|
|
|
property_name === 'realm_message_content_delete_limit_minutes') {
|
|
|
|
set_msg_delete_limit_dropdown();
|
2018-04-05 00:54:31 +02:00
|
|
|
} else if (property_name === 'realm_org_join_restrictions') {
|
|
|
|
set_org_join_restrictions_dropdown();
|
2019-01-14 14:04:08 +01:00
|
|
|
} else if (property_name === 'realm_message_content_allowed_in_email_notifications') {
|
|
|
|
set_message_content_in_email_notifications_visiblity();
|
2019-04-06 06:34:49 +02:00
|
|
|
} else if (property_name === 'realm_digest_emails_enabled') {
|
2019-05-10 08:45:46 +02:00
|
|
|
settings_notifications.set_enable_digest_emails_visibility();
|
2019-05-08 08:11:30 +02:00
|
|
|
set_digest_emails_weekday_visibility();
|
2018-03-29 12:52:57 +02:00
|
|
|
}
|
2018-04-01 08:49:10 +02:00
|
|
|
}
|
2018-03-29 12:52:57 +02:00
|
|
|
|
2018-03-29 12:58:35 +02:00
|
|
|
function discard_property_element_changes(elem) {
|
|
|
|
elem = $(elem);
|
2019-11-02 00:06:25 +01:00
|
|
|
const property_name = exports.extract_property_name(elem);
|
|
|
|
const property_value = get_property_value(property_name);
|
2018-03-29 12:58:35 +02:00
|
|
|
|
2020-01-13 10:45:41 +01:00
|
|
|
if (property_name === 'realm_authentication_methods') {
|
|
|
|
exports.populate_auth_methods(property_value);
|
2020-01-13 12:32:38 +01:00
|
|
|
} else if (property_name === 'realm_notifications_stream') {
|
|
|
|
exports.render_notifications_stream_ui(property_value, "notifications");
|
|
|
|
} else if (property_name === 'realm_signup_notifications_stream') {
|
|
|
|
exports.render_notifications_stream_ui(property_value, "signup_notifications");
|
2020-03-31 15:21:27 +02:00
|
|
|
} else if (property_name === 'realm_default_code_block_language') {
|
|
|
|
exports.default_code_language_widget.render(property_value);
|
2020-01-13 10:45:41 +01:00
|
|
|
} else if (typeof property_value === 'boolean') {
|
2018-03-29 12:58:35 +02:00
|
|
|
elem.prop('checked', property_value);
|
|
|
|
} else if (typeof property_value === 'string' || typeof property_value === 'number') {
|
|
|
|
elem.val(property_value);
|
|
|
|
} else {
|
|
|
|
blueslip.error('Element refers to unknown property ' + property_name);
|
|
|
|
}
|
|
|
|
|
2018-04-01 08:56:18 +02:00
|
|
|
update_dependent_subsettings(property_name);
|
2018-03-29 12:58:35 +02:00
|
|
|
}
|
|
|
|
|
2018-03-29 13:15:50 +02:00
|
|
|
exports.sync_realm_settings = function (property) {
|
|
|
|
if (!overlays.settings_open()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (property === 'message_content_edit_limit_seconds') {
|
|
|
|
property = 'message_content_edit_limit_minutes';
|
2018-04-02 11:18:40 +02:00
|
|
|
} else if (property === 'allow_message_editing') {
|
|
|
|
property = 'msg_edit_limit_setting';
|
2018-07-27 23:26:29 +02:00
|
|
|
} else if (property === 'emails_restricted_to_domains' || property === 'disallow_disposable_email_addresses') {
|
2018-04-05 00:54:31 +02:00
|
|
|
property = 'org_join_restrictions';
|
2017-11-26 09:12:10 +01:00
|
|
|
} else if (property === 'message_content_delete_limit_seconds') {
|
|
|
|
property = 'message_content_delete_limit_minutes';
|
|
|
|
} else if (property === 'allow_message_deleting') {
|
|
|
|
property = 'msg_delete_limit_setting';
|
2018-04-30 18:26:59 +02:00
|
|
|
} else if (property === 'invite_required' || property === 'invite_by_admins_only') {
|
|
|
|
property = 'user_invite_restriction';
|
2018-03-29 13:15:50 +02:00
|
|
|
}
|
2019-11-02 00:06:25 +01:00
|
|
|
const element = $('#id_realm_' + property);
|
2018-03-29 13:15:50 +02:00
|
|
|
if (element.length) {
|
|
|
|
discard_property_element_changes(element);
|
|
|
|
}
|
|
|
|
};
|
2019-03-07 15:18:10 +01:00
|
|
|
|
2018-03-29 13:15:50 +02:00
|
|
|
|
2018-03-29 02:02:01 +02:00
|
|
|
exports.change_save_button_state = function ($element, state) {
|
2019-03-07 15:18:10 +01:00
|
|
|
function show_hide_element($element, show, fadeout_delay) {
|
|
|
|
if (show) {
|
|
|
|
$element.removeClass('hide').addClass('.show').fadeIn(300);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
setTimeout(function () {
|
|
|
|
$element.fadeOut(300);
|
|
|
|
}, fadeout_delay);
|
|
|
|
}
|
|
|
|
|
2019-11-02 00:06:25 +01:00
|
|
|
const $saveBtn = $element.find('.save-button');
|
|
|
|
const $textEl = $saveBtn.find('.icon-button-text');
|
2019-03-07 15:18:10 +01:00
|
|
|
|
2018-03-29 02:02:01 +02:00
|
|
|
if (state !== "saving") {
|
|
|
|
$saveBtn.removeClass('saving');
|
|
|
|
}
|
2019-03-07 15:18:10 +01:00
|
|
|
|
|
|
|
if (state === "discarded") {
|
2019-08-18 14:48:40 +02:00
|
|
|
show_hide_element($element, false, 0);
|
2019-03-07 15:18:10 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-11-02 00:06:25 +01:00
|
|
|
let button_text;
|
|
|
|
let data_status;
|
|
|
|
let is_show;
|
2018-03-29 02:02:01 +02:00
|
|
|
if (state === "unsaved") {
|
2019-03-09 08:34:53 +01:00
|
|
|
button_text = i18n.t("Save changes");
|
|
|
|
data_status = "unsaved";
|
|
|
|
is_show = true;
|
2019-03-07 15:18:10 +01:00
|
|
|
|
|
|
|
$element.find('.discard-button').show();
|
2018-03-29 02:02:01 +02:00
|
|
|
} else if (state === "saved") {
|
2019-03-09 08:34:53 +01:00
|
|
|
button_text = i18n.t("Save changes");
|
|
|
|
data_status = "";
|
|
|
|
is_show = false;
|
2018-03-29 02:02:01 +02:00
|
|
|
} else if (state === "saving") {
|
2019-03-09 08:34:53 +01:00
|
|
|
button_text = i18n.t("Saving");
|
|
|
|
data_status = "saving";
|
|
|
|
is_show = true;
|
2019-03-07 15:18:10 +01:00
|
|
|
|
|
|
|
$element.find('.discard-button').hide();
|
2018-03-29 02:02:01 +02:00
|
|
|
$saveBtn.addClass('saving');
|
|
|
|
} else if (state === "failed") {
|
2019-03-09 08:34:53 +01:00
|
|
|
button_text = i18n.t("Save changes");
|
|
|
|
data_status = "failed";
|
|
|
|
is_show = true;
|
2018-03-29 02:02:01 +02:00
|
|
|
} else if (state === 'succeeded') {
|
2019-03-09 08:34:53 +01:00
|
|
|
button_text = i18n.t("Saved");
|
|
|
|
data_status = "saved";
|
|
|
|
is_show = false;
|
|
|
|
}
|
|
|
|
|
2019-03-07 15:18:10 +01:00
|
|
|
$textEl.text(button_text);
|
|
|
|
$saveBtn.attr("data-status", data_status);
|
|
|
|
show_hide_element($element, is_show, 800);
|
2018-03-29 02:02:01 +02:00
|
|
|
};
|
|
|
|
|
2020-03-22 16:13:50 +01:00
|
|
|
exports.get_input_element_value = function (input_elem) {
|
|
|
|
input_elem = $(input_elem);
|
|
|
|
const input_type = input_elem.data("setting-widget-type");
|
|
|
|
if (input_type) {
|
|
|
|
if (input_type === 'bool') {
|
|
|
|
return input_elem.prop('checked');
|
|
|
|
}
|
|
|
|
if (input_type === 'text') {
|
|
|
|
return input_elem.val().trim();
|
|
|
|
}
|
|
|
|
if (input_type === 'integer') {
|
|
|
|
return parseInt(input_elem.val().trim(), 10);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
};
|
|
|
|
|
2018-05-30 18:03:34 +02:00
|
|
|
exports.set_up = function () {
|
2018-12-08 18:16:37 +01:00
|
|
|
exports.build_page();
|
|
|
|
exports.maybe_disable_widgets();
|
|
|
|
};
|
|
|
|
|
|
|
|
exports.build_page = function () {
|
2017-04-08 18:13:39 +02:00
|
|
|
meta.loaded = true;
|
|
|
|
|
|
|
|
loading.make_indicator($('#admin_page_auth_methods_loading_indicator'));
|
|
|
|
|
2017-06-10 07:03:53 +02:00
|
|
|
// Populate notifications stream modal
|
|
|
|
if (page_params.is_admin) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const streams = stream_data.get_streams_for_settings_page();
|
2017-10-20 16:55:04 +02:00
|
|
|
exports.populate_notifications_stream_dropdown(streams);
|
|
|
|
exports.populate_signup_notifications_stream_dropdown(streams);
|
2020-03-31 15:21:27 +02:00
|
|
|
exports.default_code_language_widget.setup();
|
2017-06-10 07:03:53 +02:00
|
|
|
}
|
2020-01-11 21:55:17 +01:00
|
|
|
exports.render_notifications_stream_ui(page_params.realm_notifications_stream_id, 'notifications');
|
|
|
|
exports.render_notifications_stream_ui(page_params.realm_signup_notifications_stream_id, 'signup_notifications');
|
2020-03-31 15:21:27 +02:00
|
|
|
exports.default_code_language_widget.render(page_params.realm_default_code_block_language);
|
2017-06-10 07:03:53 +02:00
|
|
|
|
2017-04-08 18:13:39 +02:00
|
|
|
// Populate realm domains
|
2017-04-20 07:30:51 +02:00
|
|
|
exports.populate_realm_domains(page_params.realm_domains);
|
2017-04-08 18:13:39 +02:00
|
|
|
|
|
|
|
// Populate authentication methods table
|
|
|
|
exports.populate_auth_methods(page_params.realm_authentication_methods);
|
2019-04-07 18:32:42 +02:00
|
|
|
insert_tip_box();
|
2017-04-08 18:13:39 +02:00
|
|
|
|
2020-02-03 12:40:36 +01:00
|
|
|
simple_dropdown_properties.forEach(set_property_dropdown_value);
|
2017-05-25 00:50:07 +02:00
|
|
|
|
2019-05-07 19:06:05 +02:00
|
|
|
set_realm_waiting_period_dropdown();
|
2018-04-23 14:51:30 +02:00
|
|
|
set_video_chat_provider_dropdown();
|
2018-04-02 11:18:40 +02:00
|
|
|
set_msg_edit_limit_dropdown();
|
2017-11-26 09:12:10 +01:00
|
|
|
set_msg_delete_limit_dropdown();
|
2018-04-05 00:54:31 +02:00
|
|
|
set_org_join_restrictions_dropdown();
|
2019-01-14 14:04:08 +01:00
|
|
|
set_message_content_in_email_notifications_visiblity();
|
2019-05-08 08:11:30 +02:00
|
|
|
set_digest_emails_weekday_visibility();
|
2018-03-05 19:19:19 +01:00
|
|
|
|
2019-04-06 08:43:34 +02:00
|
|
|
function get_auth_method_table_data() {
|
2019-11-02 00:06:25 +01:00
|
|
|
const new_auth_methods = {};
|
|
|
|
const auth_method_rows = $("#id_realm_authentication_methods").find('tr.method_row');
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
|
|
|
|
for (const method_row of auth_method_rows) {
|
2019-04-06 08:43:34 +02:00
|
|
|
new_auth_methods[$(method_row).data('method')] = $(method_row).find('input').prop('checked');
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
}
|
|
|
|
|
2019-04-06 08:43:34 +02:00
|
|
|
return new_auth_methods;
|
|
|
|
}
|
|
|
|
|
2018-03-14 23:50:17 +01:00
|
|
|
function check_property_changed(elem) {
|
|
|
|
elem = $(elem);
|
2019-11-02 00:06:25 +01:00
|
|
|
const property_name = exports.extract_property_name(elem);
|
|
|
|
let changed_val;
|
|
|
|
let current_val = get_property_value(property_name);
|
2020-01-13 10:45:41 +01:00
|
|
|
|
|
|
|
if (property_name === 'realm_authentication_methods') {
|
|
|
|
current_val = sort_object_by_key(current_val);
|
|
|
|
current_val = JSON.stringify(current_val);
|
|
|
|
changed_val = get_auth_method_table_data();
|
|
|
|
changed_val = JSON.stringify(changed_val);
|
2020-01-13 12:32:38 +01:00
|
|
|
} else if (property_name === 'realm_notifications_stream') {
|
|
|
|
changed_val = parseInt($("#id_realm_notifications_stream").data('stream-id'), 10);
|
|
|
|
} else if (property_name === 'realm_signup_notifications_stream') {
|
|
|
|
changed_val = parseInt($("#id_realm_signup_notifications_stream").data('stream-id'), 10);
|
2020-03-31 15:21:27 +02:00
|
|
|
} else if (property_name === 'realm_default_code_block_language') {
|
|
|
|
changed_val = exports.default_code_language_widget.value();
|
2020-01-13 10:45:41 +01:00
|
|
|
} else if (typeof current_val === 'boolean') {
|
2018-03-14 23:50:17 +01:00
|
|
|
changed_val = elem.prop('checked');
|
|
|
|
} else if (typeof current_val === 'string') {
|
|
|
|
changed_val = elem.val().trim();
|
|
|
|
} else if (typeof current_val === 'number') {
|
|
|
|
current_val = current_val.toString();
|
|
|
|
changed_val = elem.val().trim();
|
|
|
|
} else {
|
2018-03-22 13:50:27 +01:00
|
|
|
blueslip.error('Element refers to unknown property ' + property_name);
|
2018-03-14 23:50:17 +01:00
|
|
|
}
|
|
|
|
return current_val !== changed_val;
|
|
|
|
}
|
|
|
|
|
2020-01-13 10:28:45 +01:00
|
|
|
function save_discard_widget_status_handler(subsection) {
|
|
|
|
subsection.find('.subsection-failed-status p').hide();
|
|
|
|
subsection.find('.save-button').show();
|
|
|
|
const properties_elements = get_subsection_property_elements(subsection);
|
2020-02-16 10:31:49 +01:00
|
|
|
const show_change_process_button = properties_elements.some(check_property_changed);
|
2020-01-13 10:28:45 +01:00
|
|
|
|
|
|
|
const save_btn_controls = subsection.find('.subsection-header .save-button-controls');
|
|
|
|
const button_state = show_change_process_button ? "unsaved" : "discarded";
|
|
|
|
exports.change_save_button_state(save_btn_controls, button_state);
|
|
|
|
}
|
|
|
|
|
2018-03-26 12:06:42 +02:00
|
|
|
$('.admin-realm-form').on('change input', 'input, select, textarea', function (e) {
|
2018-03-14 23:50:17 +01:00
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
|
|
|
|
2019-08-22 01:48:38 +02:00
|
|
|
// This event handler detects whether after these input
|
|
|
|
// changes, any fields have different values from the current
|
|
|
|
// official values stored in the database and page_params. If
|
|
|
|
// they do, we transition to the "unsaved" state showing the
|
|
|
|
// save/discard widget; otherwise, we hide that widget (the
|
|
|
|
// "discarded" state).
|
2019-08-05 21:11:30 +02:00
|
|
|
|
|
|
|
if ($(e.target).hasClass("no-input-change-detection")) {
|
|
|
|
// This is to prevent input changes detection in elements
|
|
|
|
// within a subsection whose changes should not affect the
|
|
|
|
// visibility of the discard button
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-11-02 00:06:25 +01:00
|
|
|
const subsection = $(e.target).closest('.org-subsection-parent');
|
2020-01-13 10:28:45 +01:00
|
|
|
save_discard_widget_status_handler(subsection);
|
2018-03-14 23:50:17 +01:00
|
|
|
});
|
|
|
|
|
2018-03-29 02:02:01 +02:00
|
|
|
$('.organization').on('click', '.subsection-header .subsection-changes-discard .button', function (e) {
|
2018-03-22 13:27:52 +01:00
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
2020-02-08 05:56:44 +01:00
|
|
|
get_subsection_property_elements(e.target).forEach(discard_property_element_changes);
|
2019-11-02 00:06:25 +01:00
|
|
|
const save_btn_controls = $(e.target).closest('.save-button-controls');
|
2018-03-29 02:02:01 +02:00
|
|
|
exports.change_save_button_state(save_btn_controls, "discarded");
|
2018-03-14 23:52:36 +01:00
|
|
|
});
|
|
|
|
|
2019-03-25 11:52:55 +01:00
|
|
|
exports.save_organization_settings = function (data, save_button) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const subsection_parent = save_button.closest('.org-subsection-parent');
|
|
|
|
const save_btn_container = subsection_parent.find('.save-button-controls');
|
|
|
|
const failed_alert_elem = subsection_parent.find('.subsection-failed-status p');
|
2018-03-29 02:02:01 +02:00
|
|
|
exports.change_save_button_state(save_btn_container, "saving");
|
2017-04-08 18:13:39 +02:00
|
|
|
channel.patch({
|
2018-03-14 23:45:42 +01:00
|
|
|
url: "/json/realm",
|
2017-04-08 18:13:39 +02:00
|
|
|
data: data,
|
2019-03-25 11:52:55 +01:00
|
|
|
success: function () {
|
2018-03-14 23:45:42 +01:00
|
|
|
failed_alert_elem.hide();
|
2019-03-07 15:18:10 +01:00
|
|
|
exports.change_save_button_state(save_btn_container, "succeeded");
|
2017-05-25 00:50:07 +02:00
|
|
|
},
|
|
|
|
error: function (xhr) {
|
2018-03-29 02:02:01 +02:00
|
|
|
exports.change_save_button_state(save_btn_container, "failed");
|
2018-04-30 15:27:35 +02:00
|
|
|
save_button.hide();
|
|
|
|
ui_report.error(i18n.t("Save failed"), xhr, failed_alert_elem);
|
2017-05-25 00:50:07 +02:00
|
|
|
},
|
|
|
|
});
|
2018-01-11 17:47:37 +01:00
|
|
|
};
|
|
|
|
|
2018-08-04 08:18:21 +02:00
|
|
|
exports.parse_time_limit = function parse_time_limit(elem) {
|
2018-04-28 15:35:14 +02:00
|
|
|
return Math.floor(parseFloat(elem.val(), 10).toFixed(1) * 60);
|
2018-08-04 08:18:21 +02:00
|
|
|
};
|
2018-04-28 15:35:14 +02:00
|
|
|
|
2018-03-21 00:50:00 +01:00
|
|
|
function get_complete_data_for_subsection(subsection) {
|
2019-11-02 00:06:25 +01:00
|
|
|
let data = {};
|
2020-02-21 21:45:34 +01:00
|
|
|
|
2018-03-14 23:45:42 +01:00
|
|
|
if (subsection === 'msg_editing') {
|
2019-11-02 00:06:25 +01:00
|
|
|
const edit_limit_setting_value = $("#id_realm_msg_edit_limit_setting").val();
|
2018-04-02 11:18:40 +02:00
|
|
|
if (edit_limit_setting_value === 'never') {
|
2019-03-25 11:52:55 +01:00
|
|
|
data.allow_message_editing = false;
|
2018-04-02 11:18:40 +02:00
|
|
|
} else if (edit_limit_setting_value === 'custom_limit') {
|
2018-08-04 08:18:21 +02:00
|
|
|
data.message_content_edit_limit_seconds = exports.parse_time_limit($('#id_realm_message_content_edit_limit_minutes'));
|
2018-04-28 15:35:14 +02:00
|
|
|
// Disable editing if the parsed time limit is 0 seconds
|
2019-03-25 11:52:55 +01:00
|
|
|
data.allow_message_editing = !!data.message_content_edit_limit_seconds;
|
2018-04-02 11:18:40 +02:00
|
|
|
} else {
|
2019-03-25 11:52:55 +01:00
|
|
|
data.allow_message_editing = true;
|
|
|
|
data.message_content_edit_limit_seconds =
|
2020-02-09 05:23:25 +01:00
|
|
|
settings_config.msg_edit_limit_dropdown_values.get(
|
|
|
|
edit_limit_setting_value
|
|
|
|
).seconds;
|
2018-03-14 23:45:42 +01:00
|
|
|
}
|
2019-11-02 00:06:25 +01:00
|
|
|
const delete_limit_setting_value = $("#id_realm_msg_delete_limit_setting").val();
|
2017-11-26 09:12:10 +01:00
|
|
|
if (delete_limit_setting_value === 'never') {
|
2019-03-25 11:52:55 +01:00
|
|
|
data.allow_message_deleting = false;
|
2017-11-26 09:12:10 +01:00
|
|
|
} else if (delete_limit_setting_value === 'custom_limit') {
|
2018-08-04 08:18:21 +02:00
|
|
|
data.message_content_delete_limit_seconds = exports.parse_time_limit($('#id_realm_message_content_delete_limit_minutes'));
|
2018-04-28 15:35:14 +02:00
|
|
|
// Disable deleting if the parsed time limit is 0 seconds
|
2019-03-25 11:52:55 +01:00
|
|
|
data.allow_message_deleting = !!data.message_content_delete_limit_seconds;
|
2017-11-26 09:12:10 +01:00
|
|
|
} else {
|
2019-03-25 11:52:55 +01:00
|
|
|
data.allow_message_deleting = true;
|
|
|
|
data.message_content_delete_limit_seconds =
|
2020-02-09 05:23:25 +01:00
|
|
|
settings_config.msg_delete_limit_dropdown_values.get(
|
|
|
|
delete_limit_setting_value
|
|
|
|
).seconds;
|
2017-11-26 09:12:10 +01:00
|
|
|
}
|
2020-01-13 12:32:38 +01:00
|
|
|
} else if (subsection === 'notifications') {
|
|
|
|
data.notifications_stream_id = JSON.stringify(
|
|
|
|
parseInt($('#id_realm_notifications_stream').data('stream-id'), 10));
|
|
|
|
data.signup_notifications_stream_id = JSON.stringify(
|
|
|
|
parseInt($('#id_realm_signup_notifications_stream').data('stream-id'), 10));
|
2019-04-27 07:48:03 +02:00
|
|
|
} else if (subsection === 'other_settings') {
|
2019-11-02 00:06:25 +01:00
|
|
|
let new_message_retention_days = $("#id_realm_message_retention_days").val();
|
2018-03-22 14:07:52 +01:00
|
|
|
|
2018-05-06 21:43:17 +02:00
|
|
|
if (parseInt(new_message_retention_days, 10).toString() !== new_message_retention_days
|
|
|
|
&& new_message_retention_days !== "") {
|
|
|
|
new_message_retention_days = "";
|
2018-03-22 14:07:52 +01:00
|
|
|
}
|
|
|
|
|
2020-03-31 15:21:27 +02:00
|
|
|
const code_block_language_value = exports.default_code_language_widget.value();
|
|
|
|
data.default_code_block_language = JSON.stringify(code_block_language_value);
|
|
|
|
|
2019-03-25 11:52:55 +01:00
|
|
|
data.message_retention_days = new_message_retention_days !== "" ?
|
|
|
|
JSON.stringify(parseInt(new_message_retention_days, 10)) : null;
|
2019-03-25 13:48:07 +01:00
|
|
|
} else if (subsection === 'other_permissions') {
|
2019-11-02 00:06:25 +01:00
|
|
|
const add_emoji_permission = $("#id_realm_add_emoji_by_admins_only").val();
|
2018-03-22 14:07:52 +01:00
|
|
|
|
|
|
|
if (add_emoji_permission === "by_admins_only") {
|
|
|
|
data.add_emoji_by_admins_only = true;
|
|
|
|
} else if (add_emoji_permission === "by_anyone") {
|
|
|
|
data.add_emoji_by_admins_only = false;
|
|
|
|
}
|
2018-04-05 00:54:31 +02:00
|
|
|
} else if (subsection === 'org_join') {
|
2019-11-02 00:06:25 +01:00
|
|
|
const org_join_restrictions = $('#id_realm_org_join_restrictions').val();
|
2018-04-05 00:54:31 +02:00
|
|
|
if (org_join_restrictions === "only_selected_domain") {
|
2019-03-25 11:52:55 +01:00
|
|
|
data.emails_restricted_to_domains = true;
|
|
|
|
data.disallow_disposable_email_addresses = false;
|
2018-04-05 00:54:31 +02:00
|
|
|
} else if (org_join_restrictions === "no_disposable_email") {
|
2019-03-25 11:52:55 +01:00
|
|
|
data.emails_restricted_to_domains = false;
|
|
|
|
data.disallow_disposable_email_addresses = true;
|
2018-04-05 00:54:31 +02:00
|
|
|
} else if (org_join_restrictions === "no_restriction") {
|
2019-03-25 11:52:55 +01:00
|
|
|
data.disallow_disposable_email_addresses = false;
|
|
|
|
data.emails_restricted_to_domains = false;
|
2018-04-05 00:54:31 +02:00
|
|
|
}
|
2018-04-30 18:26:59 +02:00
|
|
|
|
2019-11-02 00:06:25 +01:00
|
|
|
const user_invite_restriction = $('#id_realm_user_invite_restriction').val();
|
2018-04-30 18:26:59 +02:00
|
|
|
if (user_invite_restriction === 'no_invite_required') {
|
2019-03-25 11:52:55 +01:00
|
|
|
data.invite_required = false;
|
|
|
|
data.invite_by_admins_only = false;
|
2018-04-30 18:26:59 +02:00
|
|
|
} else if (user_invite_restriction === 'by_admins_only') {
|
2019-03-25 11:52:55 +01:00
|
|
|
data.invite_required = true;
|
|
|
|
data.invite_by_admins_only = true;
|
2018-04-30 18:26:59 +02:00
|
|
|
} else {
|
2019-03-25 11:52:55 +01:00
|
|
|
data.invite_required = true;
|
|
|
|
data.invite_by_admins_only = false;
|
2018-04-30 18:26:59 +02:00
|
|
|
}
|
2020-03-23 21:51:27 +01:00
|
|
|
|
|
|
|
const waiting_period_threshold = $("#id_realm_waiting_period_setting").val();
|
|
|
|
if (waiting_period_threshold === "none") {
|
|
|
|
data.waiting_period_threshold = 0;
|
|
|
|
} else if (waiting_period_threshold === "three_days") {
|
|
|
|
data.waiting_period_threshold = 3;
|
|
|
|
} else if (waiting_period_threshold === "custom_days") {
|
|
|
|
data.waiting_period_threshold = $("#id_realm_waiting_period_threshold").val();
|
|
|
|
}
|
2019-04-06 08:43:34 +02:00
|
|
|
} else if (subsection === 'auth_settings') {
|
|
|
|
data = {};
|
|
|
|
data.authentication_methods = JSON.stringify(get_auth_method_table_data());
|
2019-07-30 09:52:06 +02:00
|
|
|
} else if (subsection === 'user_defaults') {
|
2019-11-02 00:06:25 +01:00
|
|
|
const realm_default_twenty_four_hour_time = $('#id_realm_default_twenty_four_hour_time').val();
|
2019-07-30 09:52:06 +02:00
|
|
|
data.default_twenty_four_hour_time = realm_default_twenty_four_hour_time;
|
2018-03-14 23:45:42 +01:00
|
|
|
}
|
2019-03-25 11:52:55 +01:00
|
|
|
return data;
|
2018-03-21 00:50:00 +01:00
|
|
|
}
|
|
|
|
|
2019-05-05 15:13:52 +02:00
|
|
|
function populate_data_for_request(subsection) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const data = {};
|
|
|
|
const properties_elements = get_subsection_property_elements(subsection);
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
|
|
|
|
for (let input_elem of properties_elements) {
|
2019-05-05 15:13:52 +02:00
|
|
|
input_elem = $(input_elem);
|
2019-05-05 15:49:37 +02:00
|
|
|
if (check_property_changed(input_elem)) {
|
2020-03-22 16:13:50 +01:00
|
|
|
const input_value = exports.get_input_element_value(input_elem);
|
|
|
|
if (input_value) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const property_name = input_elem.attr('id').replace("id_realm_", "");
|
2020-03-22 16:13:50 +01:00
|
|
|
data[property_name] = JSON.stringify(input_value);
|
2019-05-05 15:13:52 +02:00
|
|
|
}
|
|
|
|
}
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
}
|
|
|
|
|
2019-05-05 15:13:52 +02:00
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2018-03-29 02:02:01 +02:00
|
|
|
$(".organization").on("click", ".subsection-header .subsection-changes-save .button", function (e) {
|
2018-03-21 00:50:00 +01:00
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
2019-11-02 00:06:25 +01:00
|
|
|
const save_button = $(e.currentTarget);
|
|
|
|
const subsection_id = save_button.attr('id').replace("org-submit-", "");
|
|
|
|
const subsection = subsection_id.split('-').join('_');
|
|
|
|
const subsection_elem = save_button.closest('.org-subsection-parent');
|
2018-03-21 00:50:00 +01:00
|
|
|
|
2020-02-09 04:15:38 +01:00
|
|
|
const data = {
|
|
|
|
...populate_data_for_request(subsection_elem),
|
|
|
|
...get_complete_data_for_subsection(subsection),
|
|
|
|
};
|
2019-03-25 11:52:55 +01:00
|
|
|
exports.save_organization_settings(data, save_button);
|
2017-05-25 00:50:07 +02:00
|
|
|
});
|
2017-08-18 01:23:55 +02:00
|
|
|
|
2018-03-22 14:16:53 +01:00
|
|
|
$(".org-subsection-parent").on("keydown", "input", function (e) {
|
|
|
|
e.stopPropagation();
|
|
|
|
if (e.keyCode === 13) {
|
|
|
|
e.preventDefault();
|
|
|
|
$(e.target).closest('.org-subsection-parent').find('.subsection-changes-save button').click();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-04-02 11:18:40 +02:00
|
|
|
$("#id_realm_msg_edit_limit_setting").change(function (e) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const msg_edit_limit_dropdown_value = e.target.value;
|
2020-02-03 13:03:53 +01:00
|
|
|
change_element_block_display_property('id_realm_message_content_edit_limit_minutes',
|
|
|
|
msg_edit_limit_dropdown_value === 'custom_limit');
|
2018-04-02 11:18:40 +02:00
|
|
|
});
|
|
|
|
|
2017-11-26 09:12:10 +01:00
|
|
|
$("#id_realm_msg_delete_limit_setting").change(function (e) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const msg_delete_limit_dropdown_value = e.target.value;
|
2020-02-03 13:03:53 +01:00
|
|
|
change_element_block_display_property('id_realm_message_content_delete_limit_minutes',
|
|
|
|
msg_delete_limit_dropdown_value === 'custom_limit');
|
2017-11-26 09:12:10 +01:00
|
|
|
});
|
|
|
|
|
2019-05-07 19:06:05 +02:00
|
|
|
$("#id_realm_waiting_period_setting").change(function () {
|
2019-11-02 00:06:25 +01:00
|
|
|
const waiting_period_threshold = this.value;
|
2020-02-03 13:03:53 +01:00
|
|
|
change_element_block_display_property('id_realm_waiting_period_threshold',
|
|
|
|
waiting_period_threshold === 'custom_days');
|
2018-01-05 15:34:10 +01:00
|
|
|
});
|
|
|
|
|
2018-04-23 14:51:30 +02:00
|
|
|
$("#id_realm_video_chat_provider").change(function (e) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const available_providers = page_params.realm_available_video_chat_providers;
|
|
|
|
const video_chat_provider_id = parseInt(e.target.value, 10);
|
2019-05-09 09:54:38 +02:00
|
|
|
|
|
|
|
if (video_chat_provider_id === available_providers.google_hangouts.id) {
|
2018-12-28 20:45:54 +01:00
|
|
|
$("#google_hangouts_domain").show();
|
|
|
|
$(".zoom_credentials").hide();
|
2019-05-09 09:54:38 +02:00
|
|
|
} else if (video_chat_provider_id === available_providers.zoom.id) {
|
2018-12-28 20:45:54 +01:00
|
|
|
$("#google_hangouts_domain").hide();
|
|
|
|
$(".zoom_credentials").show();
|
2018-04-23 14:51:30 +02:00
|
|
|
} else {
|
2018-12-28 20:45:54 +01:00
|
|
|
$("#google_hangouts_domain").hide();
|
|
|
|
$(".zoom_credentials").hide();
|
2018-04-23 14:51:30 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-04-05 00:54:31 +02:00
|
|
|
$("#id_realm_org_join_restrictions").change(function (e) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const org_join_restrictions = e.target.value;
|
|
|
|
const node = $("#allowed_domains_label").parent();
|
2018-04-05 00:54:31 +02:00
|
|
|
if (org_join_restrictions === 'only_selected_domain') {
|
|
|
|
node.show();
|
2020-02-08 06:23:59 +01:00
|
|
|
if (page_params.realm_domains.length === 0) {
|
2018-04-05 00:54:31 +02:00
|
|
|
overlays.open_modal('realm_domains_modal');
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
node.hide();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
$("#id_realm_org_join_restrictions").click(function (e) {
|
|
|
|
// This prevents the disappearance of modal when there are
|
|
|
|
// no allowed domains otherwise it gets closed due to
|
|
|
|
// the click event handler attached to `#settings_overlay_container`
|
|
|
|
e.stopPropagation();
|
|
|
|
});
|
|
|
|
|
2018-04-28 21:34:57 +02:00
|
|
|
function fade_status_element(elem) {
|
|
|
|
setTimeout(function () {
|
|
|
|
elem.fadeOut(500);
|
|
|
|
}, 1000);
|
|
|
|
}
|
|
|
|
|
2017-04-08 18:13:39 +02:00
|
|
|
$("#realm_domains_table").on("click", ".delete_realm_domain", function () {
|
2019-11-02 00:06:25 +01:00
|
|
|
const domain = $(this).parents("tr").find(".domain").text();
|
|
|
|
const url = "/json/realm/domains/" + domain;
|
|
|
|
const realm_domains_info = $(".realm_domains_info");
|
2017-04-08 18:13:39 +02:00
|
|
|
|
|
|
|
channel.del({
|
|
|
|
url: url,
|
|
|
|
success: function () {
|
2017-12-25 17:49:46 +01:00
|
|
|
ui_report.success(i18n.t("Deleted successfully!"), realm_domains_info);
|
2018-04-28 21:34:57 +02:00
|
|
|
fade_status_element(realm_domains_info);
|
2017-04-08 18:13:39 +02:00
|
|
|
},
|
|
|
|
error: function (xhr) {
|
2017-12-25 17:49:46 +01:00
|
|
|
ui_report.error(i18n.t("Failed"), xhr, realm_domains_info);
|
2018-04-28 21:34:57 +02:00
|
|
|
fade_status_element(realm_domains_info);
|
2017-04-08 18:13:39 +02:00
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
$("#submit-add-realm-domain").click(function () {
|
2019-11-02 00:06:25 +01:00
|
|
|
const realm_domains_info = $(".realm_domains_info");
|
|
|
|
const widget = $("#add-realm-domain-widget");
|
|
|
|
const domain = widget.find(".new-realm-domain").val();
|
|
|
|
const allow_subdomains = widget.find(".new-realm-domain-allow-subdomains").prop("checked");
|
|
|
|
const data = {
|
2017-04-08 18:13:39 +02:00
|
|
|
domain: JSON.stringify(domain),
|
|
|
|
allow_subdomains: JSON.stringify(allow_subdomains),
|
|
|
|
};
|
|
|
|
|
|
|
|
channel.post({
|
|
|
|
url: "/json/realm/domains",
|
|
|
|
data: data,
|
|
|
|
success: function () {
|
|
|
|
$("#add-realm-domain-widget .new-realm-domain").val("");
|
|
|
|
$("#add-realm-domain-widget .new-realm-domain-allow-subdomains").prop("checked", false);
|
2017-12-25 17:49:46 +01:00
|
|
|
ui_report.success(i18n.t("Added successfully!"), realm_domains_info);
|
2018-04-28 21:34:57 +02:00
|
|
|
fade_status_element(realm_domains_info);
|
2017-04-08 18:13:39 +02:00
|
|
|
},
|
|
|
|
error: function (xhr) {
|
2017-12-25 17:49:46 +01:00
|
|
|
ui_report.error(i18n.t("Failed"), xhr, realm_domains_info);
|
2018-04-28 21:34:57 +02:00
|
|
|
fade_status_element(realm_domains_info);
|
2017-04-08 18:13:39 +02:00
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
$("#realm_domains_table").on("change", ".allow-subdomains", function (e) {
|
|
|
|
e.stopPropagation();
|
2019-11-02 00:06:25 +01:00
|
|
|
const realm_domains_info = $(".realm_domains_info");
|
|
|
|
const domain = $(this).parents("tr").find(".domain").text();
|
|
|
|
const allow_subdomains = $(this).prop('checked');
|
|
|
|
const url = '/json/realm/domains/' + domain;
|
|
|
|
const data = {
|
2017-04-08 18:13:39 +02:00
|
|
|
allow_subdomains: JSON.stringify(allow_subdomains),
|
|
|
|
};
|
|
|
|
|
|
|
|
channel.patch({
|
|
|
|
url: url,
|
|
|
|
data: data,
|
|
|
|
success: function () {
|
|
|
|
if (allow_subdomains) {
|
2017-12-25 17:49:46 +01:00
|
|
|
ui_report.success(i18n.t("Update successful: Subdomains allowed for __domain__",
|
|
|
|
{domain: domain}), realm_domains_info);
|
2017-04-08 18:13:39 +02:00
|
|
|
} else {
|
2017-12-25 17:49:46 +01:00
|
|
|
ui_report.success(i18n.t("Update successful: Subdomains no longer allowed for __domain__",
|
|
|
|
{domain: domain}), realm_domains_info);
|
2017-04-08 18:13:39 +02:00
|
|
|
}
|
2018-04-28 21:34:57 +02:00
|
|
|
fade_status_element(realm_domains_info);
|
2017-04-08 18:13:39 +02:00
|
|
|
},
|
|
|
|
error: function (xhr) {
|
2017-12-25 17:49:46 +01:00
|
|
|
ui_report.error(i18n.t("Failed"), xhr, realm_domains_info);
|
2018-04-28 21:34:57 +02:00
|
|
|
fade_status_element(realm_domains_info);
|
2017-04-08 18:13:39 +02:00
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-01-11 21:25:52 +01:00
|
|
|
function notification_stream_update(stream_id, notification_type) {
|
2020-01-11 21:55:17 +01:00
|
|
|
exports.render_notifications_stream_ui(stream_id, notification_type);
|
2020-01-13 12:32:38 +01:00
|
|
|
save_discard_widget_status_handler($('#org-notifications'));
|
2017-10-20 16:55:04 +02:00
|
|
|
}
|
|
|
|
|
2020-03-31 15:21:27 +02:00
|
|
|
exports.default_code_language_widget.register_event_handlers(
|
|
|
|
save_discard_widget_status_handler);
|
|
|
|
|
2020-01-11 21:25:52 +01:00
|
|
|
$(".notifications-stream-setting .dropdown-list-body").on("click keypress", ".stream_name", function (e) {
|
|
|
|
const notifications_stream_setting_elem = $(this).closest(".notifications-stream-setting");
|
2017-10-20 16:55:04 +02:00
|
|
|
if (e.type === "keypress") {
|
|
|
|
if (e.which === 13) {
|
2020-01-11 21:25:52 +01:00
|
|
|
notifications_stream_setting_elem.find(".dropdown-menu").dropdown("toggle");
|
2017-10-20 16:55:04 +02:00
|
|
|
} else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2019-12-30 13:20:45 +01:00
|
|
|
const stream_id = parseInt($(this).attr('data-stream-id'), 10);
|
2020-01-11 21:25:52 +01:00
|
|
|
notification_stream_update(stream_id, notifications_stream_setting_elem.data("notifications-type"));
|
2017-10-20 16:55:04 +02:00
|
|
|
});
|
|
|
|
|
2020-01-11 21:25:52 +01:00
|
|
|
$(".notification-disable").click(function (e) {
|
|
|
|
notification_stream_update(-1, e.target.id.replace("_stream_disable", ""));
|
2017-10-20 16:55:04 +02:00
|
|
|
});
|
|
|
|
|
2017-04-08 18:13:39 +02:00
|
|
|
function upload_realm_icon(file_input) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const form_data = new FormData();
|
2017-04-08 18:13:39 +02:00
|
|
|
|
|
|
|
form_data.append('csrfmiddlewaretoken', csrf_token);
|
2020-02-08 01:53:49 +01:00
|
|
|
for (const [i, file] of Array.prototype.entries.call(file_input[0].files)) {
|
2018-06-04 21:13:07 +02:00
|
|
|
form_data.append('file-' + i, file);
|
2020-02-08 01:53:49 +01:00
|
|
|
}
|
2017-04-08 18:13:39 +02:00
|
|
|
|
2019-11-02 00:06:25 +01:00
|
|
|
const error_field = $("#realm_icon_file_input_error");
|
2018-08-16 01:26:55 +02:00
|
|
|
error_field.hide();
|
2019-11-02 00:06:25 +01:00
|
|
|
const spinner = $("#upload_icon_spinner").expectOne();
|
2019-03-09 17:43:48 +01:00
|
|
|
loading.make_indicator(spinner, {text: i18n.t("Uploading profile picture.")});
|
2018-08-16 01:26:55 +02:00
|
|
|
$("#upload_icon_button_text").expectOne().hide();
|
2017-04-08 18:13:39 +02:00
|
|
|
|
2017-07-05 19:02:54 +02:00
|
|
|
channel.post({
|
2017-04-08 18:13:39 +02:00
|
|
|
url: '/json/realm/icon',
|
|
|
|
data: form_data,
|
|
|
|
cache: false,
|
|
|
|
processData: false,
|
|
|
|
contentType: false,
|
|
|
|
success: function () {
|
|
|
|
loading.destroy_indicator($("#upload_icon_spinner"));
|
2018-08-16 01:26:55 +02:00
|
|
|
$("#upload_icon_button_text").expectOne().show();
|
|
|
|
},
|
|
|
|
error: function (xhr) {
|
2019-01-13 08:51:46 +01:00
|
|
|
loading.destroy_indicator($("#upload_icon_spinner"));
|
|
|
|
$("#upload_icon_button_text").expectOne().show();
|
2018-08-16 01:26:55 +02:00
|
|
|
ui_report.error("", xhr, error_field);
|
2017-04-08 18:13:39 +02:00
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
realm_icon.build_realm_icon_widget(upload_realm_icon);
|
|
|
|
|
2019-01-27 08:25:10 +01:00
|
|
|
function upload_realm_logo(file_input, night) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const form_data = new FormData();
|
|
|
|
let spinner;
|
|
|
|
let error_field;
|
|
|
|
let button_text;
|
2018-08-16 01:26:55 +02:00
|
|
|
|
|
|
|
form_data.append('csrfmiddlewaretoken', csrf_token);
|
2020-02-08 01:53:49 +01:00
|
|
|
for (const [i, file] of Array.prototype.entries.call(file_input[0].files)) {
|
2018-08-16 01:26:55 +02:00
|
|
|
form_data.append('file-' + i, file);
|
2020-02-08 01:53:49 +01:00
|
|
|
}
|
2019-01-27 08:25:10 +01:00
|
|
|
if (night) {
|
2019-08-21 10:37:51 +02:00
|
|
|
error_field = $("#night-logo-section .realm-logo-file-input-error");
|
2019-08-21 12:25:14 +02:00
|
|
|
spinner = $("#night-logo-section .upload-logo-spinner");
|
|
|
|
button_text = $("#night-logo-section .upload-logo-button-text");
|
2019-01-27 08:25:10 +01:00
|
|
|
} else {
|
2019-08-21 10:37:51 +02:00
|
|
|
error_field = $("#day-logo-section .realm-logo-file-input-error");
|
2019-08-21 12:25:14 +02:00
|
|
|
spinner = $("#day-logo-section .upload-logo-spinner");
|
|
|
|
button_text = $("#day-logo-section .upload-logo-button-text");
|
2019-01-27 08:25:10 +01:00
|
|
|
}
|
|
|
|
spinner.expectOne();
|
2018-08-16 01:26:55 +02:00
|
|
|
error_field.hide();
|
2019-01-27 08:25:10 +01:00
|
|
|
button_text.expectOne().hide();
|
2018-08-16 01:26:55 +02:00
|
|
|
loading.make_indicator(spinner, {text: i18n.t("Uploading logo.")});
|
2019-01-27 08:25:10 +01:00
|
|
|
form_data.append('night', JSON.stringify(night));
|
2018-08-16 01:26:55 +02:00
|
|
|
channel.post({
|
|
|
|
url: '/json/realm/logo',
|
|
|
|
data: form_data,
|
|
|
|
cache: false,
|
|
|
|
processData: false,
|
|
|
|
contentType: false,
|
|
|
|
success: function () {
|
2019-01-27 08:25:10 +01:00
|
|
|
loading.destroy_indicator(spinner);
|
|
|
|
button_text.expectOne().show();
|
2018-08-16 01:26:55 +02:00
|
|
|
},
|
|
|
|
error: function (xhr) {
|
2019-01-27 08:25:10 +01:00
|
|
|
loading.destroy_indicator(spinner);
|
|
|
|
button_text.expectOne().show();
|
2018-08-16 01:26:55 +02:00
|
|
|
ui_report.error("", xhr, error_field);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
2019-02-27 15:45:26 +01:00
|
|
|
|
2019-06-12 14:22:37 +02:00
|
|
|
if (page_params.plan_includes_wide_organization_logo) {
|
|
|
|
realm_logo.build_realm_logo_widget(upload_realm_logo, false);
|
|
|
|
realm_logo.build_realm_logo_widget(upload_realm_logo, true);
|
|
|
|
}
|
2019-03-11 16:48:59 +01:00
|
|
|
|
2018-08-16 01:26:55 +02:00
|
|
|
|
2018-01-30 14:58:50 +01:00
|
|
|
$('#deactivate_realm_button').on('click', function (e) {
|
|
|
|
if (!overlays.is_modal_open()) {
|
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
|
|
|
overlays.open_modal('deactivate-realm-modal');
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
$('#do_deactivate_realm_button').on('click', function () {
|
|
|
|
if (overlays.is_modal_open()) {
|
|
|
|
overlays.close_modal('deactivate-realm-modal');
|
|
|
|
}
|
|
|
|
channel.post({
|
2018-12-18 19:34:45 +01:00
|
|
|
url: '/json/realm/deactivate',
|
2018-01-30 14:58:50 +01:00
|
|
|
error: function (xhr) {
|
|
|
|
ui_report.error(
|
|
|
|
i18n.t("Failed"), xhr, $('#admin-realm-deactivation-status').expectOne()
|
|
|
|
);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
2017-04-08 18:13:39 +02:00
|
|
|
};
|
|
|
|
|
2019-10-25 09:45:13 +02:00
|
|
|
window.settings_org = exports;
|