zulip/static/js/settings_exports.js

146 lines
4.6 KiB
JavaScript
Raw Normal View History

import $ from "jquery";
import render_confirm_delete_data_export from "../templates/confirm_dialog/confirm_delete_data_export.hbs";
import render_admin_export_list from "../templates/settings/admin_export_list.hbs";
import * as channel from "./channel";
import * as confirm_dialog from "./confirm_dialog";
import * as dialog_widget from "./dialog_widget";
import {$t_html} from "./i18n";
import * as ListWidget from "./list_widget";
import * as loading from "./loading";
import * as people from "./people";
import * as timerender from "./timerender";
import * as ui from "./ui";
import * as ui_report from "./ui_report";
2020-08-20 21:24:06 +02:00
const meta = {
loaded: false,
};
export function reset() {
meta.loaded = false;
}
function sort_user(a, b) {
const a_name = people.get_full_name(a.acting_user_id).toLowerCase();
const b_name = people.get_full_name(b.acting_user_id).toLowerCase();
if (a_name > b_name) {
return 1;
} else if (a_name === b_name) {
return 0;
}
return -1;
}
export function populate_exports_table(exports) {
if (!meta.loaded) {
return;
}
const $exports_table = $("#admin_exports_table").expectOne();
ListWidget.create($exports_table, Object.values(exports), {
name: "admin_exports_list",
modifier(data) {
let failed_timestamp = data.failed_timestamp;
let deleted_timestamp = data.deleted_timestamp;
if (failed_timestamp !== null) {
failed_timestamp = timerender.last_seen_status_from_date(
new Date(failed_timestamp * 1000),
);
}
if (deleted_timestamp !== null) {
deleted_timestamp = timerender.last_seen_status_from_date(
new Date(deleted_timestamp * 1000),
);
}
return render_admin_export_list({
realm_export: {
id: data.id,
acting_user: people.get_full_name(data.acting_user_id),
// Convert seconds -> milliseconds
event_time: timerender.last_seen_status_from_date(
new Date(data.export_time * 1000),
),
url: data.export_url,
time_failed: failed_timestamp,
pending: data.pending,
time_deleted: deleted_timestamp,
},
});
},
filter: {
$element: $exports_table.closest(".settings-section").find(".search"),
predicate(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 people.get_full_name(item.acting_user_id).toLowerCase().includes(value);
},
onupdate() {
ui.reset_scrollbar($exports_table);
},
},
$parent_container: $("#data-exports").expectOne(),
init_sort: [sort_user],
sort_fields: {
user: sort_user,
},
$simplebar_container: $("#data-exports .progressive-table-wrapper"),
});
const $spinner = $(".export_row .export_url_spinner");
if ($spinner.length) {
loading.make_indicator($spinner);
} else {
loading.destroy_indicator($spinner);
}
}
export function set_up() {
meta.loaded = true;
$("#export-data").on("click", (e) => {
e.preventDefault();
e.stopPropagation();
const $export_status = $("#export_status");
channel.post({
url: "/json/export/realm",
success() {
ui_report.success(
$t_html({defaultMessage: "Export started. Check back in a few minutes."}),
$export_status,
4000,
);
},
error(xhr) {
ui_report.error($t_html({defaultMessage: "Export failed"}), xhr, $export_status);
},
});
});
// Do an initial population of the table
channel.get({
url: "/json/export/realm",
success(data) {
populate_exports_table(data.exports);
},
});
$(".admin_exports_table").on("click", ".delete", function (e) {
e.preventDefault();
e.stopPropagation();
const $btn = $(this);
const url = "/json/export/realm/" + encodeURIComponent($btn.attr("data-export-id"));
const html_body = render_confirm_delete_data_export();
confirm_dialog.launch({
html_heading: $t_html({defaultMessage: "Delete data export?"}),
html_body,
on_click: () => dialog_widget.submit_api_request(channel.del, url),
loading_spinner: true,
});
});
}