matrix-react-sdk
Version:
SDK for matrix.org using React
275 lines (262 loc) • 37.8 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.RoomNotifState = void 0;
exports.determineUnreadState = determineUnreadState;
exports.getRoomNotifsState = getRoomNotifsState;
exports.getUnreadNotificationCount = getUnreadNotificationCount;
exports.isRuleMaybeRoomMuteRule = isRuleMaybeRoomMuteRule;
exports.setRoomNotifsState = setRoomNotifsState;
var _pushprocessor = require("matrix-js-sdk/src/pushprocessor");
var _matrix = require("matrix-js-sdk/src/matrix");
var _NotificationLevel = require("./stores/notifications/NotificationLevel");
var _RoomStatusBar = require("./components/structures/RoomStatusBar");
var _Unread = require("./Unread");
var _membership = require("./utils/membership");
var _SettingsStore = _interopRequireDefault(require("./settings/SettingsStore"));
var _notifications = require("./utils/notifications");
/*
Copyright 2024 New Vector Ltd.
Copyright 2016-2019 , 2023 The Matrix.org Foundation C.I.C.
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
let RoomNotifState = exports.RoomNotifState = /*#__PURE__*/function (RoomNotifState) {
RoomNotifState["AllMessagesLoud"] = "all_messages_loud";
RoomNotifState["AllMessages"] = "all_messages";
RoomNotifState["MentionsOnly"] = "mentions_only";
RoomNotifState["Mute"] = "mute";
return RoomNotifState;
}({});
function getRoomNotifsState(client, roomId) {
if (client.isGuest()) return RoomNotifState.AllMessages;
// look through the override rules for a rule affecting this room:
// if one exists, it will take precedence.
const muteRule = findOverrideMuteRule(client, roomId);
if (muteRule) {
return RoomNotifState.Mute;
}
// for everything else, look at the room rule.
let roomRule;
try {
roomRule = client.getRoomPushRule("global", roomId);
} catch (err) {
// Possible that the client doesn't have pushRules yet. If so, it
// hasn't started either, so indicate that this room is not notifying.
return null;
}
// XXX: We have to assume the default is to notify for all messages
// (in particular this will be 'wrong' for one to one rooms because
// they will notify loudly for all messages)
if (!roomRule?.enabled) return RoomNotifState.AllMessages;
// a mute at the room level will still allow mentions
// to notify
if (isMuteRule(roomRule)) return RoomNotifState.MentionsOnly;
const actionsObject = _pushprocessor.PushProcessor.actionListToActionsObject(roomRule.actions);
if (actionsObject.tweaks.sound) return RoomNotifState.AllMessagesLoud;
return null;
}
function setRoomNotifsState(client, roomId, newState) {
if (newState === RoomNotifState.Mute) {
return setRoomNotifsStateMuted(client, roomId);
} else {
return setRoomNotifsStateUnmuted(client, roomId, newState);
}
}
function getUnreadNotificationCount(room, type, includeThreads, threadId) {
const getCountShownForRoom = (r, type) => {
return includeThreads ? r.getUnreadNotificationCount(type) : r.getRoomUnreadNotificationCount(type);
};
let notificationCount = !!threadId ? room.getThreadUnreadNotificationCount(threadId, type) : getCountShownForRoom(room, type);
// Check notification counts in the old room just in case there's some lost
// there. We only go one level down to avoid performance issues, and theory
// is that 1st generation rooms will have already been read by the 3rd generation.
const msc3946ProcessDynamicPredecessor = _SettingsStore.default.getValue("feature_dynamic_room_predecessors");
const predecessor = room.findPredecessor(msc3946ProcessDynamicPredecessor);
// Exclude threadId, as the same thread can't continue over a room upgrade
if (!threadId && predecessor?.roomId) {
const oldRoomId = predecessor.roomId;
const oldRoom = room.client.getRoom(oldRoomId);
if (oldRoom) {
// We only ever care if there's highlights in the old room. No point in
// notifying the user for unread messages because they would have extreme
// difficulty changing their notification preferences away from "All Messages"
// and "Noisy".
notificationCount += getCountShownForRoom(oldRoom, _matrix.NotificationCountType.Highlight);
}
}
return notificationCount;
}
function setRoomNotifsStateMuted(cli, roomId) {
const promises = [];
// delete the room rule
const roomRule = cli.getRoomPushRule("global", roomId);
if (roomRule) {
promises.push(cli.deletePushRule("global", _matrix.PushRuleKind.RoomSpecific, roomRule.rule_id));
}
// add/replace an override rule to squelch everything in this room
// NB. We use the room ID as the name of this rule too, although this
// is an override rule, not a room rule: it still pertains to this room
// though, so using the room ID as the rule ID is logical and prevents
// duplicate copies of the rule.
promises.push(cli.addPushRule("global", _matrix.PushRuleKind.Override, roomId, {
conditions: [{
kind: _matrix.ConditionKind.EventMatch,
key: "room_id",
pattern: roomId
}],
actions: [_matrix.PushRuleActionName.DontNotify]
}));
return Promise.all(promises);
}
function setRoomNotifsStateUnmuted(cli, roomId, newState) {
const promises = [];
const overrideMuteRule = findOverrideMuteRule(cli, roomId);
if (overrideMuteRule) {
promises.push(cli.deletePushRule("global", _matrix.PushRuleKind.Override, overrideMuteRule.rule_id));
}
if (newState === RoomNotifState.AllMessages) {
const roomRule = cli.getRoomPushRule("global", roomId);
if (roomRule) {
promises.push(cli.deletePushRule("global", _matrix.PushRuleKind.RoomSpecific, roomRule.rule_id));
}
} else if (newState === RoomNotifState.MentionsOnly) {
promises.push(cli.addPushRule("global", _matrix.PushRuleKind.RoomSpecific, roomId, {
actions: [_matrix.PushRuleActionName.DontNotify]
}));
} else if (newState === RoomNotifState.AllMessagesLoud) {
promises.push(cli.addPushRule("global", _matrix.PushRuleKind.RoomSpecific, roomId, {
actions: [_matrix.PushRuleActionName.Notify, {
set_tweak: _matrix.TweakName.Sound,
value: "default"
}]
}));
}
return Promise.all(promises);
}
function findOverrideMuteRule(cli, roomId) {
if (!cli?.pushRules?.global?.override) {
return null;
}
for (const rule of cli.pushRules.global.override) {
if (rule.enabled && isRuleRoomMuteRuleForRoomId(roomId, rule)) {
return rule;
}
}
return null;
}
/**
* Checks if a given rule is a room mute rule as implemented by EW
* - matches every event in one room (one condition that is an event match on roomId)
* - silences notifications (one action that is `DontNotify`)
* @param rule - push rule
* @returns {boolean} - true when rule mutes a room
*/
function isRuleMaybeRoomMuteRule(rule) {
return (
// matches every event in one room
rule.conditions?.length === 1 && rule.conditions[0].kind === _matrix.ConditionKind.EventMatch && rule.conditions[0].key === "room_id" &&
// silences notifications
isMuteRule(rule)
);
}
/**
* Checks if a given rule is a room mute rule as implemented by EW
* @param roomId - id of room to match
* @param rule - push rule
* @returns {boolean} true when rule mutes the given room
*/
function isRuleRoomMuteRuleForRoomId(roomId, rule) {
if (!isRuleMaybeRoomMuteRule(rule)) {
return false;
}
// isRuleMaybeRoomMuteRule checks this condition exists
const cond = rule.conditions[0];
return cond.pattern === roomId;
}
function isMuteRule(rule) {
// DontNotify is equivalent to the empty actions array
return rule.actions.length === 0 || rule.actions.length === 1 && rule.actions[0] === _matrix.PushRuleActionName.DontNotify;
}
/**
* Returns an object giving information about the unread state of a room or thread
* @param room The room to query, or the room the thread is in
* @param threadId The thread to check the unread state of, or undefined to query the main thread
* @param includeThreads If threadId is undefined, true to include threads other than the main thread, or
* false to exclude them. Ignored if threadId is specified.
* @returns
*/
function determineUnreadState(room, threadId, includeThreads) {
if (!room) {
return {
symbol: null,
count: 0,
level: _NotificationLevel.NotificationLevel.None
};
}
if ((0, _RoomStatusBar.getUnsentMessages)(room, threadId).length > 0) {
return {
symbol: "!",
count: 1,
level: _NotificationLevel.NotificationLevel.Unsent
};
}
if ((0, _membership.getEffectiveMembership)(room.getMyMembership()) === _membership.EffectiveMembership.Invite) {
return {
symbol: "!",
count: 1,
level: _NotificationLevel.NotificationLevel.Highlight
};
}
if (_SettingsStore.default.getValue("feature_ask_to_join") && (0, _membership.isKnockDenied)(room)) {
return {
symbol: "!",
count: 1,
level: _NotificationLevel.NotificationLevel.Highlight
};
}
if (getRoomNotifsState(room.client, room.roomId) === RoomNotifState.Mute) {
return {
symbol: null,
count: 0,
level: _NotificationLevel.NotificationLevel.None
};
}
const redNotifs = getUnreadNotificationCount(room, _matrix.NotificationCountType.Highlight, includeThreads ?? false, threadId);
const greyNotifs = getUnreadNotificationCount(room, _matrix.NotificationCountType.Total, includeThreads ?? false, threadId);
const trueCount = greyNotifs || redNotifs;
if (redNotifs > 0) {
return {
symbol: null,
count: trueCount,
level: _NotificationLevel.NotificationLevel.Highlight
};
}
const markedUnreadState = (0, _notifications.getMarkedUnreadState)(room);
if (greyNotifs > 0 || markedUnreadState) {
return {
symbol: null,
count: trueCount,
level: _NotificationLevel.NotificationLevel.Notification
};
}
// We don't have any notified messages, but we might have unread messages. Let's find out.
let hasUnread = false;
if (threadId) {
const thread = room.getThread(threadId);
if (thread) {
hasUnread = (0, _Unread.doesRoomOrThreadHaveUnreadMessages)(thread);
}
// If the thread does not exist, assume it contains no unreads
} else {
hasUnread = (0, _Unread.doesRoomHaveUnreadMessages)(room, includeThreads ?? false);
}
return {
symbol: null,
count: trueCount,
level: hasUnread ? _NotificationLevel.NotificationLevel.Activity : _NotificationLevel.NotificationLevel.None
};
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfcHVzaHByb2Nlc3NvciIsInJlcXVpcmUiLCJfbWF0cml4IiwiX05vdGlmaWNhdGlvbkxldmVsIiwiX1Jvb21TdGF0dXNCYXIiLCJfVW5yZWFkIiwiX21lbWJlcnNoaXAiLCJfU2V0dGluZ3NTdG9yZSIsIl9pbnRlcm9wUmVxdWlyZURlZmF1bHQiLCJfbm90aWZpY2F0aW9ucyIsIlJvb21Ob3RpZlN0YXRlIiwiZXhwb3J0cyIsImdldFJvb21Ob3RpZnNTdGF0ZSIsImNsaWVudCIsInJvb21JZCIsImlzR3Vlc3QiLCJBbGxNZXNzYWdlcyIsIm11dGVSdWxlIiwiZmluZE92ZXJyaWRlTXV0ZVJ1bGUiLCJNdXRlIiwicm9vbVJ1bGUiLCJnZXRSb29tUHVzaFJ1bGUiLCJlcnIiLCJlbmFibGVkIiwiaXNNdXRlUnVsZSIsIk1lbnRpb25zT25seSIsImFjdGlvbnNPYmplY3QiLCJQdXNoUHJvY2Vzc29yIiwiYWN0aW9uTGlzdFRvQWN0aW9uc09iamVjdCIsImFjdGlvbnMiLCJ0d2Vha3MiLCJzb3VuZCIsIkFsbE1lc3NhZ2VzTG91ZCIsInNldFJvb21Ob3RpZnNTdGF0ZSIsIm5ld1N0YXRlIiwic2V0Um9vbU5vdGlmc1N0YXRlTXV0ZWQiLCJzZXRSb29tTm90aWZzU3RhdGVVbm11dGVkIiwiZ2V0VW5yZWFkTm90aWZpY2F0aW9uQ291bnQiLCJyb29tIiwidHlwZSIsImluY2x1ZGVUaHJlYWRzIiwidGhyZWFkSWQiLCJnZXRDb3VudFNob3duRm9yUm9vbSIsInIiLCJnZXRSb29tVW5yZWFkTm90aWZpY2F0aW9uQ291bnQiLCJub3RpZmljYXRpb25Db3VudCIsImdldFRocmVhZFVucmVhZE5vdGlmaWNhdGlvbkNvdW50IiwibXNjMzk0NlByb2Nlc3NEeW5hbWljUHJlZGVjZXNzb3IiLCJTZXR0aW5nc1N0b3JlIiwiZ2V0VmFsdWUiLCJwcmVkZWNlc3NvciIsImZpbmRQcmVkZWNlc3NvciIsIm9sZFJvb21JZCIsIm9sZFJvb20iLCJnZXRSb29tIiwiTm90aWZpY2F0aW9uQ291bnRUeXBlIiwiSGlnaGxpZ2h0IiwiY2xpIiwicHJvbWlzZXMiLCJwdXNoIiwiZGVsZXRlUHVzaFJ1bGUiLCJQdXNoUnVsZUtpbmQiLCJSb29tU3BlY2lmaWMiLCJydWxlX2lkIiwiYWRkUHVzaFJ1bGUiLCJPdmVycmlkZSIsImNvbmRpdGlvbnMiLCJraW5kIiwiQ29uZGl0aW9uS2luZCIsIkV2ZW50TWF0Y2giLCJrZXkiLCJwYXR0ZXJuIiwiUHVzaFJ1bGVBY3Rpb25OYW1lIiwiRG9udE5vdGlmeSIsIlByb21pc2UiLCJhbGwiLCJvdmVycmlkZU11dGVSdWxlIiwiTm90aWZ5Iiwic2V0X3R3ZWFrIiwiVHdlYWtOYW1lIiwiU291bmQiLCJ2YWx1ZSIsInB1c2hSdWxlcyIsImdsb2JhbCIsIm92ZXJyaWRlIiwicnVsZSIsImlzUnVsZVJvb21NdXRlUnVsZUZvclJvb21JZCIsImlzUnVsZU1heWJlUm9vbU11dGVSdWxlIiwibGVuZ3RoIiwiY29uZCIsImRldGVybWluZVVucmVhZFN0YXRlIiwic3ltYm9sIiwiY291bnQiLCJsZXZlbCIsIk5vdGlmaWNhdGlvbkxldmVsIiwiTm9uZSIsImdldFVuc2VudE1lc3NhZ2VzIiwiVW5zZW50IiwiZ2V0RWZmZWN0aXZlTWVtYmVyc2hpcCIsImdldE15TWVtYmVyc2hpcCIsIkVmZmVjdGl2ZU1lbWJlcnNoaXAiLCJJbnZpdGUiLCJpc0tub2NrRGVuaWVkIiwicmVkTm90aWZzIiwiZ3JleU5vdGlmcyIsIlRvdGFsIiwidHJ1ZUNvdW50IiwibWFya2VkVW5yZWFkU3RhdGUiLCJnZXRNYXJrZWRVbnJlYWRTdGF0ZSIsIk5vdGlmaWNhdGlvbiIsImhhc1VucmVhZCIsInRocmVhZCIsImdldFRocmVhZCIsImRvZXNSb29tT3JUaHJlYWRIYXZlVW5yZWFkTWVzc2FnZXMiLCJkb2VzUm9vbUhhdmVVbnJlYWRNZXNzYWdlcyIsIkFjdGl2aXR5Il0sInNvdXJjZXMiOlsiLi4vc3JjL1Jvb21Ob3RpZnMudHMiXSwic291cmNlc0NvbnRlbnQiOlsiLypcbkNvcHlyaWdodCAyMDI0IE5ldyBWZWN0b3IgTHRkLlxuQ29weXJpZ2h0IDIwMTYtMjAxOSAsIDIwMjMgVGhlIE1hdHJpeC5vcmcgRm91bmRhdGlvbiBDLkkuQy5cblxuU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IEFHUEwtMy4wLW9ubHkgT1IgR1BMLTMuMC1vbmx5XG5QbGVhc2Ugc2VlIExJQ0VOU0UgZmlsZXMgaW4gdGhlIHJlcG9zaXRvcnkgcm9vdCBmb3IgZnVsbCBkZXRhaWxzLlxuKi9cblxuaW1wb3J0IHsgUHVzaFByb2Nlc3NvciB9IGZyb20gXCJtYXRyaXgtanMtc2RrL3NyYy9wdXNocHJvY2Vzc29yXCI7XG5pbXBvcnQge1xuICAgIE5vdGlmaWNhdGlvbkNvdW50VHlwZSxcbiAgICBDb25kaXRpb25LaW5kLFxuICAgIFB1c2hSdWxlQWN0aW9uTmFtZSxcbiAgICBQdXNoUnVsZUtpbmQsXG4gICAgVHdlYWtOYW1lLFxufSBmcm9tIFwibWF0cml4LWpzLXNkay9zcmMvbWF0cml4XCI7XG5cbmltcG9ydCB0eXBlIHsgSVB1c2hSdWxlLCBSb29tLCBNYXRyaXhDbGllbnQgfSBmcm9tIFwibWF0cml4LWpzLXNkay9zcmMvbWF0cml4XCI7XG5pbXBvcnQgeyBOb3RpZmljYXRpb25MZXZlbCB9IGZyb20gXCIuL3N0b3Jlcy9ub3RpZmljYXRpb25zL05vdGlmaWNhdGlvbkxldmVsXCI7XG5pbXBvcnQgeyBnZXRVbnNlbnRNZXNzYWdlcyB9IGZyb20gXCIuL2NvbXBvbmVudHMvc3RydWN0dXJlcy9Sb29tU3RhdHVzQmFyXCI7XG5pbXBvcnQgeyBkb2VzUm9vbUhhdmVVbnJlYWRNZXNzYWdlcywgZG9lc1Jvb21PclRocmVhZEhhdmVVbnJlYWRNZXNzYWdlcyB9IGZyb20gXCIuL1VucmVhZFwiO1xuaW1wb3J0IHsgRWZmZWN0aXZlTWVtYmVyc2hpcCwgZ2V0RWZmZWN0aXZlTWVtYmVyc2hpcCwgaXNLbm9ja0RlbmllZCB9IGZyb20gXCIuL3V0aWxzL21lbWJlcnNoaXBcIjtcbmltcG9ydCBTZXR0aW5nc1N0b3JlIGZyb20gXCIuL3NldHRpbmdzL1NldHRpbmdzU3RvcmVcIjtcbmltcG9ydCB7IGdldE1hcmtlZFVucmVhZFN0YXRlIH0gZnJvbSBcIi4vdXRpbHMvbm90aWZpY2F0aW9uc1wiO1xuXG5leHBvcnQgZW51bSBSb29tTm90aWZTdGF0ZSB7XG4gICAgQWxsTWVzc2FnZXNMb3VkID0gXCJhbGxfbWVzc2FnZXNfbG91ZFwiLFxuICAgIEFsbE1lc3NhZ2VzID0gXCJhbGxfbWVzc2FnZXNcIixcbiAgICBNZW50aW9uc09ubHkgPSBcIm1lbnRpb25zX29ubHlcIixcbiAgICBNdXRlID0gXCJtdXRlXCIsXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRSb29tTm90aWZzU3RhdGUoY2xpZW50OiBNYXRyaXhDbGllbnQsIHJvb21JZDogc3RyaW5nKTogUm9vbU5vdGlmU3RhdGUgfCBudWxsIHtcbiAgICBpZiAoY2xpZW50LmlzR3Vlc3QoKSkgcmV0dXJuIFJvb21Ob3RpZlN0YXRlLkFsbE1lc3NhZ2VzO1xuXG4gICAgLy8gbG9vayB0aHJvdWdoIHRoZSBvdmVycmlkZSBydWxlcyBmb3IgYSBydWxlIGFmZmVjdGluZyB0aGlzIHJvb206XG4gICAgLy8gaWYgb25lIGV4aXN0cywgaXQgd2lsbCB0YWtlIHByZWNlZGVuY2UuXG4gICAgY29uc3QgbXV0ZVJ1bGUgPSBmaW5kT3ZlcnJpZGVNdXRlUnVsZShjbGllbnQsIHJvb21JZCk7XG4gICAgaWYgKG11dGVSdWxlKSB7XG4gICAgICAgIHJldHVybiBSb29tTm90aWZTdGF0ZS5NdXRlO1xuICAgIH1cblxuICAgIC8vIGZvciBldmVyeXRoaW5nIGVsc2UsIGxvb2sgYXQgdGhlIHJvb20gcnVsZS5cbiAgICBsZXQgcm9vbVJ1bGU6IElQdXNoUnVsZSB8IHVuZGVmaW5lZDtcbiAgICB0cnkge1xuICAgICAgICByb29tUnVsZSA9IGNsaWVudC5nZXRSb29tUHVzaFJ1bGUoXCJnbG9iYWxcIiwgcm9vbUlkKTtcbiAgICB9IGNhdGNoIChlcnIpIHtcbiAgICAgICAgLy8gUG9zc2libGUgdGhhdCB0aGUgY2xpZW50IGRvZXNuJ3QgaGF2ZSBwdXNoUnVsZXMgeWV0LiBJZiBzbywgaXRcbiAgICAgICAgLy8gaGFzbid0IHN0YXJ0ZWQgZWl0aGVyLCBzbyBpbmRpY2F0ZSB0aGF0IHRoaXMgcm9vbSBpcyBub3Qgbm90aWZ5aW5nLlxuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICAvLyBYWFg6IFdlIGhhdmUgdG8gYXNzdW1lIHRoZSBkZWZhdWx0IGlzIHRvIG5vdGlmeSBmb3IgYWxsIG1lc3NhZ2VzXG4gICAgLy8gKGluIHBhcnRpY3VsYXIgdGhpcyB3aWxsIGJlICd3cm9uZycgZm9yIG9uZSB0byBvbmUgcm9vbXMgYmVjYXVzZVxuICAgIC8vIHRoZXkgd2lsbCBub3RpZnkgbG91ZGx5IGZvciBhbGwgbWVzc2FnZXMpXG4gICAgaWYgKCFyb29tUnVsZT8uZW5hYmxlZCkgcmV0dXJuIFJvb21Ob3RpZlN0YXRlLkFsbE1lc3NhZ2VzO1xuXG4gICAgLy8gYSBtdXRlIGF0IHRoZSByb29tIGxldmVsIHdpbGwgc3RpbGwgYWxsb3cgbWVudGlvbnNcbiAgICAvLyB0byBub3RpZnlcbiAgICBpZiAoaXNNdXRlUnVsZShyb29tUnVsZSkpIHJldHVybiBSb29tTm90aWZTdGF0ZS5NZW50aW9uc09ubHk7XG5cbiAgICBjb25zdCBhY3Rpb25zT2JqZWN0ID0gUHVzaFByb2Nlc3Nvci5hY3Rpb25MaXN0VG9BY3Rpb25zT2JqZWN0KHJvb21SdWxlLmFjdGlvbnMpO1xuICAgIGlmIChhY3Rpb25zT2JqZWN0LnR3ZWFrcy5zb3VuZCkgcmV0dXJuIFJvb21Ob3RpZlN0YXRlLkFsbE1lc3NhZ2VzTG91ZDtcblxuICAgIHJldHVybiBudWxsO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gc2V0Um9vbU5vdGlmc1N0YXRlKGNsaWVudDogTWF0cml4Q2xpZW50LCByb29tSWQ6IHN0cmluZywgbmV3U3RhdGU6IFJvb21Ob3RpZlN0YXRlKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgaWYgKG5ld1N0YXRlID09PSBSb29tTm90aWZTdGF0ZS5NdXRlKSB7XG4gICAgICAgIHJldHVybiBzZXRSb29tTm90aWZzU3RhdGVNdXRlZChjbGllbnQsIHJvb21JZCk7XG4gICAgfSBlbHNlIHtcbiAgICAgICAgcmV0dXJuIHNldFJvb21Ob3RpZnNTdGF0ZVVubXV0ZWQoY2xpZW50LCByb29tSWQsIG5ld1N0YXRlKTtcbiAgICB9XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRVbnJlYWROb3RpZmljYXRpb25Db3VudChcbiAgICByb29tOiBSb29tLFxuICAgIHR5cGU6IE5vdGlmaWNhdGlvbkNvdW50VHlwZSxcbiAgICBpbmNsdWRlVGhyZWFkczogYm9vbGVhbixcbiAgICB0aHJlYWRJZD86IHN0cmluZyxcbik6IG51bWJlciB7XG4gICAgY29uc3QgZ2V0Q291bnRTaG93bkZvclJvb20gPSAocjogUm9vbSwgdHlwZTogTm90aWZpY2F0aW9uQ291bnRUeXBlKTogbnVtYmVyID0+IHtcbiAgICAgICAgcmV0dXJuIGluY2x1ZGVUaHJlYWRzID8gci5nZXRVbnJlYWROb3RpZmljYXRpb25Db3VudCh0eXBlKSA6IHIuZ2V0Um9vbVVucmVhZE5vdGlmaWNhdGlvbkNvdW50KHR5cGUpO1xuICAgIH07XG5cbiAgICBsZXQgbm90aWZpY2F0aW9uQ291bnQgPSAhIXRocmVhZElkXG4gICAgICAgID8gcm9vbS5nZXRUaHJlYWRVbnJlYWROb3RpZmljYXRpb25Db3VudCh0aHJlYWRJZCwgdHlwZSlcbiAgICAgICAgOiBnZXRDb3VudFNob3duRm9yUm9vbShyb29tLCB0eXBlKTtcblxuICAgIC8vIENoZWNrIG5vdGlmaWNhdGlvbiBjb3VudHMgaW4gdGhlIG9sZCByb29tIGp1c3QgaW4gY2FzZSB0aGVyZSdzIHNvbWUgbG9zdFxuICAgIC8vIHRoZXJlLiBXZSBvbmx5IGdvIG9uZSBsZXZlbCBkb3duIHRvIGF2b2lkIHBlcmZvcm1hbmNlIGlzc3VlcywgYW5kIHRoZW9yeVxuICAgIC8vIGlzIHRoYXQgMXN0IGdlbmVyYXRpb24gcm9vbXMgd2lsbCBoYXZlIGFscmVhZHkgYmVlbiByZWFkIGJ5IHRoZSAzcmQgZ2VuZXJhdGlvbi5cbiAgICBjb25zdCBtc2MzOTQ2UHJvY2Vzc0R5bmFtaWNQcmVkZWNlc3NvciA9IFNldHRpbmdzU3RvcmUuZ2V0VmFsdWUoXCJmZWF0dXJlX2R5bmFtaWNfcm9vbV9wcmVkZWNlc3NvcnNcIik7XG4gICAgY29uc3QgcHJlZGVjZXNzb3IgPSByb29tLmZpbmRQcmVkZWNlc3Nvcihtc2MzOTQ2UHJvY2Vzc0R5bmFtaWNQcmVkZWNlc3Nvcik7XG4gICAgLy8gRXhjbHVkZSB0aHJlYWRJZCwgYXMgdGhlIHNhbWUgdGhyZWFkIGNhbid0IGNvbnRpbnVlIG92ZXIgYSByb29tIHVwZ3JhZGVcbiAgICBpZiAoIXRocmVhZElkICYmIHByZWRlY2Vzc29yPy5yb29tSWQpIHtcbiAgICAgICAgY29uc3Qgb2xkUm9vbUlkID0gcHJlZGVjZXNzb3Iucm9vbUlkO1xuICAgICAgICBjb25zdCBvbGRSb29tID0gcm9vbS5jbGllbnQuZ2V0Um9vbShvbGRSb29tSWQpO1xuICAgICAgICBpZiAob2xkUm9vbSkge1xuICAgICAgICAgICAgLy8gV2Ugb25seSBldmVyIGNhcmUgaWYgdGhlcmUncyBoaWdobGlnaHRzIGluIHRoZSBvbGQgcm9vbS4gTm8gcG9pbnQgaW5cbiAgICAgICAgICAgIC8vIG5vdGlmeWluZyB0aGUgdXNlciBmb3IgdW5yZWFkIG1lc3NhZ2VzIGJlY2F1c2UgdGhleSB3b3VsZCBoYXZlIGV4dHJlbWVcbiAgICAgICAgICAgIC8vIGRpZmZpY3VsdHkgY2hhbmdpbmcgdGhlaXIgbm90aWZpY2F0aW9uIHByZWZlcmVuY2VzIGF3YXkgZnJvbSBcIkFsbCBNZXNzYWdlc1wiXG4gICAgICAgICAgICAvLyBhbmQgXCJOb2lzeVwiLlxuICAgICAgICAgICAgbm90aWZpY2F0aW9uQ291bnQgKz0gZ2V0Q291bnRTaG93bkZvclJvb20ob2xkUm9vbSwgTm90aWZpY2F0aW9uQ291bnRUeXBlLkhpZ2hsaWdodCk7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICByZXR1cm4gbm90aWZpY2F0aW9uQ291bnQ7XG59XG5cbmZ1bmN0aW9uIHNldFJvb21Ob3RpZnNTdGF0ZU11dGVkKGNsaTogTWF0cml4Q2xpZW50LCByb29tSWQ6IHN0cmluZyk6IFByb21pc2U8YW55PiB7XG4gICAgY29uc3QgcHJvbWlzZXM6IFByb21pc2U8dW5rbm93bj5bXSA9IFtdO1xuXG4gICAgLy8gZGVsZXRlIHRoZSByb29tIHJ1bGVcbiAgICBjb25zdCByb29tUnVsZSA9IGNsaS5nZXRSb29tUHVzaFJ1bGUoXCJnbG9iYWxcIiwgcm9vbUlkKTtcbiAgICBpZiAocm9vbVJ1bGUpIHtcbiAgICAgICAgcHJvbWlzZXMucHVzaChjbGkuZGVsZXRlUHVzaFJ1bGUoXCJnbG9iYWxcIiwgUHVzaFJ1bGVLaW5kLlJvb21TcGVjaWZpYywgcm9vbVJ1bGUucnVsZV9pZCkpO1xuICAgIH1cblxuICAgIC8vIGFkZC9yZXBsYWNlIGFuIG92ZXJyaWRlIHJ1bGUgdG8gc3F1ZWxjaCBldmVyeXRoaW5nIGluIHRoaXMgcm9vbVxuICAgIC8vIE5CLiBXZSB1c2UgdGhlIHJvb20gSUQgYXMgdGhlIG5hbWUgb2YgdGhpcyBydWxlIHRvbywgYWx0aG91Z2ggdGhpc1xuICAgIC8vIGlzIGFuIG92ZXJyaWRlIHJ1bGUsIG5vdCBhIHJvb20gcnVsZTogaXQgc3RpbGwgcGVydGFpbnMgdG8gdGhpcyByb29tXG4gICAgLy8gdGhvdWdoLCBzbyB1c2luZyB0aGUgcm9vbSBJRCBhcyB0aGUgcnVsZSBJRCBpcyBsb2dpY2FsIGFuZCBwcmV2ZW50c1xuICAgIC8vIGR1cGxpY2F0ZSBjb3BpZXMgb2YgdGhlIHJ1bGUuXG4gICAgcHJvbWlzZXMucHVzaChcbiAgICAgICAgY2xpLmFkZFB1c2hSdWxlKFwiZ2xvYmFsXCIsIFB1c2hSdWxlS2luZC5PdmVycmlkZSwgcm9vbUlkLCB7XG4gICAgICAgICAgICBjb25kaXRpb25zOiBbXG4gICAgICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgICAgICBraW5kOiBDb25kaXRpb25LaW5kLkV2ZW50TWF0Y2gsXG4gICAgICAgICAgICAgICAgICAgIGtleTogXCJyb29tX2lkXCIsXG4gICAgICAgICAgICAgICAgICAgIHBhdHRlcm46IHJvb21JZCxcbiAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgXSxcbiAgICAgICAgICAgIGFjdGlvbnM6IFtQdXNoUnVsZUFjdGlvbk5hbWUuRG9udE5vdGlmeV0sXG4gICAgICAgIH0pLFxuICAgICk7XG5cbiAgICByZXR1cm4gUHJvbWlzZS5hbGwocHJvbWlzZXMpO1xufVxuXG5mdW5jdGlvbiBzZXRSb29tTm90aWZzU3RhdGVVbm11dGVkKGNsaTogTWF0cml4Q2xpZW50LCByb29tSWQ6IHN0cmluZywgbmV3U3RhdGU6IFJvb21Ob3RpZlN0YXRlKTogUHJvbWlzZTxhbnk+IHtcbiAgICBjb25zdCBwcm9taXNlczogUHJvbWlzZTx1bmtub3duPltdID0gW107XG5cbiAgICBjb25zdCBvdmVycmlkZU11dGVSdWxlID0gZmluZE92ZXJyaWRlTXV0ZVJ1bGUoY2xpLCByb29tSWQpO1xuICAgIGlmIChvdmVycmlkZU11dGVSdWxlKSB7XG4gICAgICAgIHByb21pc2VzLnB1c2goY2xpLmRlbGV0ZVB1c2hSdWxlKFwiZ2xvYmFsXCIsIFB1c2hSdWxlS2luZC5PdmVycmlkZSwgb3ZlcnJpZGVNdXRlUnVsZS5ydWxlX2lkKSk7XG4gICAgfVxuXG4gICAgaWYgKG5ld1N0YXRlID09PSBSb29tTm90aWZTdGF0ZS5BbGxNZXNzYWdlcykge1xuICAgICAgICBjb25zdCByb29tUnVsZSA9IGNsaS5nZXRSb29tUHVzaFJ1bGUoXCJnbG9iYWxcIiwgcm9vbUlkKTtcbiAgICAgICAgaWYgKHJvb21SdWxlKSB7XG4gICAgICAgICAgICBwcm9taXNlcy5wdXNoKGNsaS5kZWxldGVQdXNoUnVsZShcImdsb2JhbFwiLCBQdXNoUnVsZUtpbmQuUm9vbVNwZWNpZmljLCByb29tUnVsZS5ydWxlX2lkKSk7XG4gICAgICAgIH1cbiAgICB9IGVsc2UgaWYgKG5ld1N0YXRlID09PSBSb29tTm90aWZTdGF0ZS5NZW50aW9uc09ubHkpIHtcbiAgICAgICAgcHJvbWlzZXMucHVzaChcbiAgICAgICAgICAgIGNsaS5hZGRQdXNoUnVsZShcImdsb2JhbFwiLCBQdXNoUnVsZUtpbmQuUm9vbVNwZWNpZmljLCByb29tSWQsIHtcbiAgICAgICAgICAgICAgICBhY3Rpb25zOiBbUHVzaFJ1bGVBY3Rpb25OYW1lLkRvbnROb3RpZnldLFxuICAgICAgICAgICAgfSksXG4gICAgICAgICk7XG4gICAgfSBlbHNlIGlmIChuZXdTdGF0ZSA9PT0gUm9vbU5vdGlmU3RhdGUuQWxsTWVzc2FnZXNMb3VkKSB7XG4gICAgICAgIHByb21pc2VzLnB1c2goXG4gICAgICAgICAgICBjbGkuYWRkUHVzaFJ1bGUoXCJnbG9iYWxcIiwgUHVzaFJ1bGVLaW5kLlJvb21TcGVjaWZpYywgcm9vbUlkLCB7XG4gICAgICAgICAgICAgICAgYWN0aW9uczogW1xuICAgICAgICAgICAgICAgICAgICBQdXNoUnVsZUFjdGlvbk5hbWUuTm90aWZ5LFxuICAgICAgICAgICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgICAgICAgICBzZXRfdHdlYWs6IFR3ZWFrTmFtZS5Tb3VuZCxcbiAgICAgICAgICAgICAgICAgICAgICAgIHZhbHVlOiBcImRlZmF1bHRcIixcbiAgICAgICAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgICBdLFxuICAgICAgICAgICAgfSksXG4gICAgICAgICk7XG4gICAgfVxuXG4gICAgcmV0dXJuIFByb21pc2UuYWxsKHByb21pc2VzKTtcbn1cblxuZnVuY3Rpb24gZmluZE92ZXJyaWRlTXV0ZVJ1bGUoY2xpOiBNYXRyaXhDbGllbnQgfCB1bmRlZmluZWQsIHJvb21JZDogc3RyaW5nKTogSVB1c2hSdWxlIHwgbnVsbCB7XG4gICAgaWYgKCFjbGk/LnB1c2hSdWxlcz8uZ2xvYmFsPy5vdmVycmlkZSkge1xuICAgICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG4gICAgZm9yIChjb25zdCBydWxlIG9mIGNsaS5wdXNoUnVsZXMuZ2xvYmFsLm92ZXJyaWRlKSB7XG4gICAgICAgIGlmIChydWxlLmVuYWJsZWQgJiYgaXNSdWxlUm9vbU11dGVSdWxlRm9yUm9vbUlkKHJvb21JZCwgcnVsZSkpIHtcbiAgICAgICAgICAgIHJldHVybiBydWxlO1xuICAgICAgICB9XG4gICAgfVxuICAgIHJldHVybiBudWxsO1xufVxuXG4vKipcbiAqIENoZWNrcyBpZiBhIGdpdmVuIHJ1bGUgaXMgYSByb29tIG11dGUgcnVsZSBhcyBpbXBsZW1lbnRlZCBieSBFV1xuICogLSBtYXRjaGVzIGV2ZXJ5IGV2ZW50IGluIG9uZSByb29tIChvbmUgY29uZGl0aW9uIHRoYXQgaXMgYW4gZXZlbnQgbWF0Y2ggb24gcm9vbUlkKVxuICogLSBzaWxlbmNlcyBub3RpZmljYXRpb25zIChvbmUgYWN0aW9uIHRoYXQgaXMgYERvbnROb3RpZnlgKVxuICogQHBhcmFtIHJ1bGUgLSBwdXNoIHJ1bGVcbiAqIEByZXR1cm5zIHtib29sZWFufSAtIHRydWUgd2hlbiBydWxlIG11dGVzIGEgcm9vbVxuICovXG5leHBvcnQgZnVuY3Rpb24gaXNSdWxlTWF5YmVSb29tTXV0ZVJ1bGUocnVsZTogSVB1c2hSdWxlKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIChcbiAgICAgICAgLy8gbWF0Y2hlcyBldmVyeSBldmVudCBpbiBvbmUgcm9vbVxuICAgICAgICBydWxlLmNvbmRpdGlvbnM/Lmxlbmd0aCA9PT0gMSAmJlxuICAgICAgICBydWxlLmNvbmRpdGlvbnNbMF0ua2luZCA9PT0gQ29uZGl0aW9uS2luZC5FdmVudE1hdGNoICYmXG4gICAgICAgIHJ1bGUuY29uZGl0aW9uc1swXS5rZXkgPT09IFwicm9vbV9pZFwiICYmXG4gICAgICAgIC8vIHNpbGVuY2VzIG5vdGlmaWNhdGlvbnNcbiAgICAgICAgaXNNdXRlUnVsZShydWxlKVxuICAgICk7XG59XG5cbi8qKlxuICogQ2hlY2tzIGlmIGEgZ2l2ZW4gcnVsZSBpcyBhIHJvb20gbXV0ZSBydWxlIGFzIGltcGxlbWVudGVkIGJ5IEVXXG4gKiBAcGFyYW0gcm9vbUlkIC0gaWQgb2Ygcm9vbSB0byBtYXRjaFxuICogQHBhcmFtIHJ1bGUgLSBwdXNoIHJ1bGVcbiAqIEByZXR1cm5zIHtib29sZWFufSB0cnVlIHdoZW4gcnVsZSBtdXRlcyB0aGUgZ2l2ZW4gcm9vbVxuICovXG5mdW5jdGlvbiBpc1J1bGVSb29tTXV0ZVJ1bGVGb3JSb29tSWQocm9vbUlkOiBzdHJpbmcsIHJ1bGU6IElQdXNoUnVsZSk6IGJvb2xlYW4ge1xuICAgIGlmICghaXNSdWxlTWF5YmVSb29tTXV0ZVJ1bGUocnVsZSkpIHtcbiAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgIH1cbiAgICAvLyBpc1J1bGVNYXliZVJvb21NdXRlUnVsZSBjaGVja3MgdGhpcyBjb25kaXRpb24gZXhpc3RzXG4gICAgY29uc3QgY29uZCA9IHJ1bGUuY29uZGl0aW9ucyFbMF0hO1xuICAgIHJldHVybiBjb25kLnBhdHRlcm4gPT09IHJvb21JZDtcbn1cblxuZnVuY3Rpb24gaXNNdXRlUnVsZShydWxlOiBJUHVzaFJ1bGUpOiBib29sZWFuIHtcbiAgICAvLyBEb250Tm90aWZ5IGlzIGVxdWl2YWxlbnQgdG8gdGhlIGVtcHR5IGFjdGlvbnMgYXJyYXlcbiAgICByZXR1cm4gKFxuICAgICAgICBydWxlLmFjdGlvbnMubGVuZ3RoID09PSAwIHx8IChydWxlLmFjdGlvbnMubGVuZ3RoID09PSAxICYmIHJ1bGUuYWN0aW9uc1swXSA9PT0gUHVzaFJ1bGVBY3Rpb25OYW1lLkRvbnROb3RpZnkpXG4gICAgKTtcbn1cblxuLyoqXG4gKiBSZXR1cm5zIGFuIG9iamVjdCBnaXZpbmcgaW5mb3JtYXRpb24gYWJvdXQgdGhlIHVucmVhZCBzdGF0ZSBvZiBhIHJvb20gb3IgdGhyZWFkXG4gKiBAcGFyYW0gcm9vbSBUaGUgcm9vbSB0byBxdWVyeSwgb3IgdGhlIHJvb20gdGhlIHRocmVhZCBpcyBpblxuICogQHBhcmFtIHRocmVhZElkIFRoZSB0aHJlYWQgdG8gY2hlY2sgdGhlIHVucmVhZCBzdGF0ZSBvZiwgb3IgdW5kZWZpbmVkIHRvIHF1ZXJ5IHRoZSBtYWluIHRocmVhZFxuICogQHBhcmFtIGluY2x1ZGVUaHJlYWRzIElmIHRocmVhZElkIGlzIHVuZGVmaW5lZCwgdHJ1ZSB0byBpbmNsdWRlIHRocmVhZHMgb3RoZXIgdGhhbiB0aGUgbWFpbiB0aHJlYWQsIG9yXG4gKiAgIGZhbHNlIHRvIGV4Y2x1ZGUgdGhlbS4gSWdub3JlZCBpZiB0aHJlYWRJZCBpcyBzcGVjaWZpZWQuXG4gKiBAcmV0dXJuc1xuICovXG5leHBvcnQgZnVuY3Rpb24gZGV0ZXJtaW5lVW5yZWFkU3RhdGUoXG4gICAgcm9vbT86IFJvb20sXG4gICAgdGhyZWFkSWQ/OiBzdHJpbmcsXG4gICAgaW5jbHVkZVRocmVhZHM/OiBib29sZWFuLFxuKTogeyBsZXZlbDogTm90aWZpY2F0aW9uTGV2ZWw7IHN5bWJvbDogc3RyaW5nIHwgbnVsbDsgY291bnQ6IG51bWJlciB9IHtcbiAgICBpZiAoIXJvb20pIHtcbiAgICAgICAgcmV0dXJuIHsgc3ltYm9sOiBudWxsLCBjb3VudDogMCwgbGV2ZWw6IE5vdGlmaWNhdGlvbkxldmVsLk5vbmUgfTtcbiAgICB9XG5cbiAgICBpZiAoZ2V0VW5zZW50TWVzc2FnZXMocm9vbSwgdGhyZWFkSWQpLmxlbmd0aCA+IDApIHtcbiAgICAgICAgcmV0dXJuIHsgc3ltYm9sOiBcIiFcIiwgY291bnQ6IDEsIGxldmVsOiBOb3RpZmljYXRpb25MZXZlbC5VbnNlbnQgfTtcbiAgICB9XG5cbiAgICBpZiAoZ2V0RWZmZWN0aXZlTWVtYmVyc2hpcChyb29tLmdldE15TWVtYmVyc2hpcCgpKSA9PT0gRWZmZWN0aXZlTWVtYmVyc2hpcC5JbnZpdGUpIHtcbiAgICAgICAgcmV0dXJuIHsgc3ltYm9sOiBcIiFcIiwgY291bnQ6IDEsIGxldmVsOiBOb3RpZmljYXRpb25MZXZlbC5IaWdobGlnaHQgfTtcbiAgICB9XG5cbiAgICBpZiAoU2V0dGluZ3NTdG9yZS5nZXRWYWx1ZShcImZlYXR1cmVfYXNrX3RvX2pvaW5cIikgJiYgaXNLbm9ja0RlbmllZChyb29tKSkge1xuICAgICAgICByZXR1cm4geyBzeW1ib2w6IFwiIVwiLCBjb3VudDogMSwgbGV2ZWw6IE5vdGlmaWNhdGlvbkxldmVsLkhpZ2hsaWdodCB9O1xuICAgIH1cblxuICAgIGlmIChnZXRSb29tTm90aWZzU3RhdGUocm9vbS5jbGllbnQsIHJvb20ucm9vbUlkKSA9PT0gUm9vbU5vdGlmU3RhdGUuTXV0ZSkge1xuICAgICAgICByZXR1cm4geyBzeW1ib2w6IG51bGwsIGNvdW50OiAwLCBsZXZlbDogTm90aWZpY2F0aW9uTGV2ZWwuTm9uZSB9O1xuICAgIH1cblxuICAgIGNvbnN0IHJlZE5vdGlmcyA9IGdldFVucmVhZE5vdGlmaWNhdGlvbkNvdW50KFxuICAgICAgICByb29tLFxuICAgICAgICBOb3RpZmljYXRpb25Db3VudFR5cGUuSGlnaGxpZ2h0LFxuICAgICAgICBpbmNsdWRlVGhyZWFkcyA/PyBmYWxzZSxcbiAgICAgICAgdGhyZWFkSWQsXG4gICAgKTtcbiAgICBjb25zdCBncmV5Tm90aWZzID0gZ2V0VW5yZWFkTm90aWZpY2F0aW9uQ291bnQocm9vbSwgTm90aWZpY2F0aW9uQ291bnRUeXBlLlRvdGFsLCBpbmNsdWRlVGhyZWFkcyA/PyBmYWxzZSwgdGhyZWFkSWQpO1xuXG4gICAgY29uc3QgdHJ1ZUNvdW50ID0gZ3JleU5vdGlmcyB8fCByZWROb3RpZnM7XG4gICAgaWYgKHJlZE5vdGlmcyA+IDApIHtcbiAgICAgICAgcmV0dXJuIHsgc3ltYm9sOiBudWxsLCBjb3VudDogdHJ1ZUNvdW50LCBsZXZlbDogTm90aWZpY2F0aW9uTGV2ZWwuSGlnaGxpZ2h0IH07XG4gICAgfVxuXG4gICAgY29uc3QgbWFya2VkVW5yZWFkU3RhdGUgPSBnZXRNYXJrZWRVbnJlYWRTdGF0ZShyb29tKTtcbiAgICBpZiAoZ3JleU5vdGlmcyA+IDAgfHwgbWFya2VkVW5yZWFkU3RhdGUpIHtcbiAgICAgICAgcmV0dXJuIHsgc3ltYm9sOiBudWxsLCBjb3VudDogdHJ1ZUNvdW50LCBsZXZlbDogTm90aWZpY2F0aW9uTGV2ZWwuTm90aWZpY2F0aW9uIH07XG4gICAgfVxuXG4gICAgLy8gV2UgZG9uJ3QgaGF2ZSBhbnkgbm90aWZpZWQgbWVzc2FnZXMsIGJ1dCB3ZSBtaWdodCBoYXZlIHVucmVhZCBtZXNzYWdlcy4gTGV0J3MgZmluZCBvdXQuXG4gICAgbGV0IGhhc1VucmVhZCA9IGZhbHNlO1xuICAgIGlmICh0aHJlYWRJZCkge1xuICAgICAgICBjb25zdCB0aHJlYWQgPSByb29tLmdldFRocmVhZCh0aHJlYWRJZCk7XG4gICAgICAgIGlmICh0aHJlYWQpIHtcbiAgICAgICAgICAgIGhhc1VucmVhZCA9IGRvZXNSb29tT3JUaHJlYWRIYXZlVW5yZWFkTWVzc2FnZXModGhyZWFkKTtcbiAgICAgICAgfVxuICAgICAgICAvLyBJZiB0aGUgdGhyZWFkIGRvZXMgbm90IGV4aXN0LCBhc3N1bWUgaXQgY29udGFpbnMgbm8gdW5yZWFkc1xuICAgIH0gZWxzZSB7XG4gICAgICAgIGhhc1VucmVhZCA9IGRvZXNSb29tSGF2ZVVucmVhZE1lc3NhZ2VzKHJvb20sIGluY2x1ZGVUaHJlYWRzID8/IGZhbHNlKTtcbiAgICB9XG5cbiAgICByZXR1cm4ge1xuICAgICAgICBzeW1ib2w6IG51bGwsXG4gICAgICAgIGNvdW50OiB0cnVlQ291bnQsXG4gICAgICAgIGxldmVsOiBoYXNVbnJlYWQgPyBOb3RpZmljYXRpb25MZXZlbC5BY3Rpdml0eSA6IE5vdGlmaWNhdGlvbkxldmVsLk5vbmUsXG4gICAgfTtcbn1cbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7O0FBUUEsSUFBQUEsY0FBQSxHQUFBQyxPQUFBO0FBQ0EsSUFBQUMsT0FBQSxHQUFBRCxPQUFBO0FBU0EsSUFBQUUsa0JBQUEsR0FBQUYsT0FBQTtBQUNBLElBQUFHLGNBQUEsR0FBQUgsT0FBQTtBQUNBLElBQUFJLE9BQUEsR0FBQUosT0FBQTtBQUNBLElBQUFLLFdBQUEsR0FBQUwsT0FBQTtBQUNBLElBQUFN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