From b5f59bca8279fbecd3d90c9efaf2b495312c4ee6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Sun, 8 Mar 2026 00:18:22 -0500 Subject: [PATCH 01/14] first pass of 4.x --- languages/en_GB/poll.json | 10 +- lib/config.js | 42 +- lib/hooks.js | 98 +- lib/nodebb.js | 1 - lib/poll.js | 241 +- lib/scheduler.js | 52 - lib/serializer.js | 4 - lib/sockets.js | 86 +- lib/vote.js | 2 +- library.js | 17 +- package-lock.json | 5585 ++++--------------------------- package.json | 4 +- plugin.json | 9 +- polls-todo.md | 13 + public/js/admin.js | 35 +- public/js/poll/creator.js | 381 ++- public/js/poll/main.js | 58 +- public/js/poll/serializer.js | 173 - public/js/poll/view.js | 55 +- schema.md | 24 + templates/poll/creator.tpl | 32 +- templates/poll/manage.tpl | 18 + templates/poll/option-input.tpl | 5 + templates/poll/view.tpl | 6 +- templates/poll/view/details.tpl | 13 +- upgrades/poll-4.0-upgrade.js | 125 + 26 files changed, 1420 insertions(+), 5669 deletions(-) delete mode 100755 lib/scheduler.js delete mode 100755 lib/serializer.js create mode 100644 polls-todo.md delete mode 100644 public/js/poll/serializer.js create mode 100644 schema.md create mode 100644 templates/poll/manage.tpl create mode 100644 templates/poll/option-input.tpl create mode 100644 upgrades/poll-4.0-upgrade.js diff --git a/languages/en_GB/poll.json b/languages/en_GB/poll.json index 66d98c1..32cb87c 100755 --- a/languages/en_GB/poll.json +++ b/languages/en_GB/poll.json @@ -13,7 +13,10 @@ "settings": "Settings", "save": "Save", "reset": "Reset", - + "manage-polls": "Manage polls", + "add-poll": "Add poll", + "no-polls": "No polls found", + "confirm-remove": "Are you sure you want to remove this poll?", "creator_title": "Create a poll", "poll_title": "Poll Title", "poll_title_placeholder": "Enter poll title", @@ -24,11 +27,12 @@ "auto_end_placeholder": "Click to enter date and time", "auto_end_help": "Leaving this empty will never end the poll.", - "error.max_options": "You can only create %d options.", + "error.max_options": "You can only create %1 options.", "error.no_options": "Create at least one option.", "error.valid_date": "Please enter a valid date.", "error.not_main": "Can only add poll in main post.", + "error.invalid-post": "Invalid post.", "error.privilege.create": "You're not allowed to create a poll", "error.anon-voting-not-allowed": "This poll does not allow voting anonymously", @@ -45,7 +49,7 @@ "voting_update_disallowed_message": "You have already voted and changing your vote is not allowed for this poll", "vote_is_final": "Changing your vote is not allowed for this poll so your vote is final", - "vote_count": "users voted for this option", + "x-users-voted-for-this-option": "%1 user(s) voted for this option", "votes": "votes", "x-votes": "%1 votes", "total-votes-x": "Total Votes: %1", diff --git a/lib/config.js b/lib/config.js index 124e80a..c8488ce 100755 --- a/lib/config.js +++ b/lib/config.js @@ -1,6 +1,6 @@ 'use strict'; -const NodeBB = require('./nodebb'); +const meta = require.main.require('./src/meta'); const packageInfo = require('../package.json'); const pluginInfo = require('../plugin.json'); @@ -18,35 +18,19 @@ Config.plugin = { }; Config.defaults = { - toggles: { - allowAnon: false, - }, - limits: { - maxOptions: 10, - }, - defaults: { - title: 'Poll', - maxvotes: 1, - disallowVoteUpdate: 0, - allowAnonVoting: 0, - end: 0, - }, + defaultTitle: 'Poll', + maxOptions: 10, + maximumVotesPerUser: 1, + allowGuestsToViewResults: 0, + disallowVoteUpdate: 0, + allowAnonVoting: 0, }; -Config.settings = {}; - -Config.init = function () { - return new Promise((resolve) => { - Config.settings = new NodeBB.Settings(Config.plugin.id, Config.plugin.version, Config.defaults, resolve); - }); -}; - -Config.adminSockets = { - sync: function () { - Config.settings.sync(); - }, - getDefaults: function (socket, data, callback) { - callback(null, Config.settings.createDefaultWrapper()); - }, +Config.getSettings = async function () { + const settings = await meta.settings.get(Config.plugin.id); + return { + ...Config.defaults, + ...settings, + }; }; diff --git a/lib/hooks.js b/lib/hooks.js index b7bb0a9..73487a6 100755 --- a/lib/hooks.js +++ b/lib/hooks.js @@ -1,34 +1,54 @@ 'use strict'; -const NodeBB = require('./nodebb'); +const db = require.main.require('./src/database'); +const topics = require.main.require('./src/topics'); +const posts = require.main.require('./src/posts'); +const privileges = require.main.require('./src/privileges'); + const Config = require('./config'); const Poll = require('./poll'); -const Serializer = require('./serializer'); const Hooks = exports; Hooks.filter = {}; Hooks.action = {}; -Hooks.filter.parseRaw = function (raw) { - return Serializer.removeMarkup(raw, '[Poll]'); +Hooks.filter.configGet = async function (config) { + config.poll = await Config.getSettings(); + return config; }; -Hooks.filter.postCreate = async function (obj) { - if (Serializer.hasMarkup(obj.post.content) && obj.data.isMain) { - return await savePoll(obj); +Hooks.filter.postCreate = async function (hookData) { + // post is going to be saved to db, data is what is submitted by user + const { post, data } = hookData; + if (Array.isArray(data?.polls) && data.polls.length) { + await Poll.add(post, data.polls); } - return obj; + return hookData; +}; + +Hooks.filter.postGetFields = async function (hookData) { + if (!hookData.fields.includes('pollIds')) { + // when a new post is created it doesnt send pollIds to the client + // force load pollIds so it is available when the client needs to load the poll data + const postData = await db.getObjectsFields(hookData.pids.map(pid => `post:${pid}`), ['pollIds']); + hookData.posts.forEach((post, index) => { + if (post) { + post.pollIds = postData[index].pollIds || ''; + } + }); + } + return hookData; }; Hooks.filter.postEdit = async function (obj) { - const { tid, pollId } = await NodeBB.Posts.getPostFields(obj.data.pid, ['tid', 'pollId']); - if (pollId || !Serializer.hasMarkup(obj.post.content)) { + const { tid, pollId } = await posts.getPostFields(obj.data.pid, ['tid', 'pollId']); + if (pollId) { return obj; } - const result = await NodeBB.Topics.getTopicFields(tid, ['mainPid', 'cid']); + const result = await topics.getTopicFields(tid, ['mainPid', 'cid']); if (parseInt(result.mainPid, 10) !== parseInt(obj.data.pid, 10)) { return obj; } @@ -51,67 +71,43 @@ Hooks.filter.postEdit = async function (obj) { } // NodeBB only updates the edited, editor and content fields, so we add the pollId field manually. - await NodeBB.Posts.setPostField(obj.data.pid, 'pollId', postData.pollId); + await posts.setPostField(obj.data.pid, 'pollId', postData.pollId); return obj; }; Hooks.filter.topicPost = async function (data) { - if (Serializer.hasMarkup(data.content)) { + if (Array.isArray(data.polls) && data.polls.length) { await canCreate(data.cid, data.uid); - return data; } return data; }; -Hooks.action.postDelete = async function (data) { - const pollId = await Poll.getPollIdByPid(data.post.pid); - if (pollId) { - await Poll.delete(pollId); - } -}; - -Hooks.action.postRestore = async function (data) { - const pollId = await Poll.getPollIdByPid(data.post.pid); - if (pollId) { - await Poll.restore(pollId); +Hooks.filter.topicReply = async function (data) { + if (Array.isArray(data.polls) && data.polls.length) { + const cid = await topics.getTopicField(data.tid, 'cid'); + await canCreate(cid, data.uid); } + return data; }; -Hooks.action.topicDelete = async function (data) { - const pollId = await Poll.getPollIdByTid(data.topic.tid); - if (pollId) { - Poll.delete(pollId); +Hooks.action.postDelete = async function (data) { + const pollIds = await Poll.getPollIdsByPid(data.post.pid); + if (pollIds.length) { + await Poll.delete(pollIds); } }; -Hooks.action.topicRestore = async function (data) { - const pollId = await Poll.getPollIdByTid(data.topic.tid); - if (pollId) { - Poll.restore(pollId); +Hooks.action.postRestore = async function (data) { + const pollIds = await Poll.getPollIdsByPid(data.post.pid); + if (pollIds.length) { + await Poll.restore(pollIds); } }; async function canCreate(cid, uid) { - const can = await NodeBB.Privileges.categories.can('poll:create', cid, uid); + const can = await privileges.categories.can('poll:create', cid, uid); if (!can) { throw new Error('[[poll:error.privilege.create]]'); } } -// called from filter:post.edit and filter:post.create -async function savePoll(obj) { - const postobj = obj.post ? obj.post : obj; - const pollData = Serializer.serialize(postobj.content, Config.settings.get()); - - if (!pollData || !pollData.options.length) { - return obj; - } - - const pollId = await Poll.add(pollData, postobj); - - postobj.pollId = pollId; - postobj.content = Serializer.removeMarkup(postobj.content); - - return obj; -} - diff --git a/lib/nodebb.js b/lib/nodebb.js index f3a70d5..da0e838 100755 --- a/lib/nodebb.js +++ b/lib/nodebb.js @@ -10,7 +10,6 @@ module.exports = { Privileges: require.main.require('./src/privileges'), Plugins: require.main.require('./src/plugins'), PluginSockets: require.main.require('./src/socket.io/plugins'), - AdminSockets: require.main.require('./src/socket.io/admin').plugins, SocketIndex: require.main.require('./src/socket.io/index'), Translator: require.main.require('./src/translator'), winston: require.main.require('winston'), diff --git a/lib/poll.js b/lib/poll.js index 811e069..95fdde2 100755 --- a/lib/poll.js +++ b/lib/poll.js @@ -1,111 +1,145 @@ 'use strict'; const NodeBB = require('./nodebb'); +const db = require.main.require('./src/database'); const Vote = require('./vote'); -const Scheduler = require('./scheduler'); const Poll = exports; -const POLL_DATA_VERSION = 2; - -Poll.add = async function (pollData, postData) { - const pollId = await NodeBB.db.incrObjectField('global', 'nextPollId'); - - const poll = { - pollId: pollId, - uid: postData.editor || postData.uid, - tid: postData.tid, - pid: postData.pid, - deleted: 0, - ended: 0, - timestamp: postData.edited || postData.timestamp, - version: POLL_DATA_VERSION, - }; +Poll.add = async function (postData, polls) { + // postData is what is being saved to db + // polls is submited by user + const savePolls = []; + for (const pollData of polls) { + // eslint-disable-next-line no-await-in-loop + const pollId = await db.incrObjectField('global', 'nextPollId'); + + const poll = { + pollId: pollId, + title: pollData.title, + uid: postData.editor || postData.uid, + pid: postData.pid, + deleted: 0, + end: pollData.end || 0, + maximumVotesPerUser: pollData.maximumVotesPerUser || 1, + timestamp: postData.edited || postData.timestamp, + }; + if (Object.hasOwn(pollData, 'allowAnonVoting')) { + poll.allowAnonVoting = pollData.allowAnonVoting; + } + if (Object.hasOwn(pollData, 'disallowVoteUpdate')) { + poll.disallowVoteUpdate = pollData.disallowVoteUpdate; + } + // [{id: '1772931625035', title: 'option 1'} {id: '1772931626635', title: 'option 2'}] + poll.options = JSON.stringify(pollData.options || []); + + savePolls.push(poll); + } - pollData.options = pollData.options.map((val, i) => ({ - id: i, - title: val, - })); + if (savePolls.length) { + postData.pollIds = JSON.stringify(savePolls.map(p => p.pollId)); + } await Promise.all([ - NodeBB.db.setObject(`poll:${pollId}`, poll), - NodeBB.db.setObject(`poll:${pollId}:settings`, pollData.settings), - NodeBB.db.listAppend('polls', pollId), - NodeBB.db.setObjectField(`topic:${poll.tid}`, 'pollId', pollId), - NodeBB.db.setAdd(`poll:${pollId}:options`, pollData.options.map(option => option.id)), - NodeBB.db.setObjectBulk(pollData.options.map( - option => ([`poll:${pollId}:options:${option.id}`, option]) - )), + db.setObjectBulk(savePolls.map(p => [`poll:${p.pollId}`, p])), + db.sortedSetAdd('polls:createtime', savePolls.map(p => p.timestamp), savePolls.map(p => p.pollId)), ]); - - // Check if this poll is scheduled to end - if (parseInt(pollData.settings.end, 10) > 0) { - Poll.schedule(pollId); - } - - return pollId; }; Poll.get = async function (pollId, uid, withVotes) { - const [info, options, settings, hasVoted, vote] = await Promise.all([ - Poll.getInfo(pollId), - Poll.getOptions(pollId, withVotes), - Poll.getSettings(pollId), + const [info, hasVoted, vote] = await Promise.all([ + Poll.getInfo(pollId, withVotes), uid ? Vote.hasUidVoted(uid, pollId) : false, uid ? Vote.getUidVote(uid, pollId) : null, ]); - options.forEach((option) => { + info.options.forEach((option) => { const percentage = ((option.voteCount / info.voteCount) * 100).toFixed(0); option.percentage = isNaN(percentage) ? 0 : percentage; }); - settings.disallowVoteUpdate = parseInt(settings.disallowVoteUpdate, 10); + info.disallowVoteUpdate = parseInt(info.disallowVoteUpdate, 10); return { info, - options, - settings, + options: info.options, hasVoted, vote, }; }; -Poll.getPollIdByTid = async function (tid) { - return await NodeBB.db.getObjectField(`topic:${tid}`, 'pollId'); -}; - -Poll.getPollIdByPid = async function (pid) { - return await NodeBB.db.getObjectField(`post:${pid}`, 'pollId'); +Poll.getPollIdsByPid = async function (pid) { + const pollIdsJson = await NodeBB.db.getObjectField(`post:${pid}`, 'pollIds'); + if (!pollIdsJson) { + return []; + } + try { + return JSON.parse(pollIdsJson || '[]'); + } catch (err) { + return []; + } }; -Poll.getInfo = async function (pollId) { +Poll.getInfo = async function (pollId, withVotes = false) { const [poll, voteCount] = await Promise.all([ - NodeBB.db.getObject(`poll:${pollId}`), + db.getObject(`poll:${pollId}`), Poll.getVoteCount(pollId), ]); poll.voteCount = parseInt(voteCount, 10) || 0; + const end = parseInt(poll.end, 10); + poll.ended = end > 0 && Date.now() > end; + poll.options = await loadOptions(pollId, poll.options, withVotes); return poll; }; Poll.getOptions = async function (pollId, withVotes = false) { - const options = await NodeBB.db.getSetMembers(`poll:${pollId}:options`); - // sort options according to option id. option id are supposed to preseve the order as in Poll.add(), - // when the user create the poll - options.sort((a, b) => a - b); - return await Promise.all( - options.map(option => Poll.getOption(pollId, option, withVotes)) - ); + const optionsJson = await db.getObjectField(`poll:${pollId}`, 'options'); + return await loadOptions(pollId, optionsJson, withVotes); +}; + +Poll.getPollOptionIds = async function (pollId) { + const optionsJson = await db.getObjectField(`poll:${pollId}`, 'options'); + return tryParseOptions(optionsJson).map(opts => opts && String(opts.id)); }; +async function loadOptions(pollId, optionsJson, withVotes = false) { + const options = tryParseOptions(optionsJson); + + const votes = await db.getSortedSetsMembers(options.map(o => `poll:${pollId}:options:${o.id}:votes`)); + + options.forEach((option, index) => { + if (option) { + option.voteCount = votes[index].length; + if (withVotes) { + option.votes = votes[index]; + } + } + }); + return options; +} + +function tryParseOptions(optionsJson) { + if (!optionsJson) return []; + try { + return JSON.parse(optionsJson || '[]'); + } catch (err) { + return []; + } +} + Poll.getOption = async function (pollId, option, withVotes = false) { - const [optionData, votes, voteCount] = await Promise.all([ - NodeBB.db.getObject(`poll:${pollId}:options:${option}`), + const [optionsJson, votes, voteCount] = await Promise.all([ + db.getObjectField(`poll:${pollId}`, 'options'), withVotes ? - NodeBB.db.getSortedSetRange(`poll:${pollId}:options:${option}:votes`, 0, -1) : + db.getSortedSetRange(`poll:${pollId}:options:${option}:votes`, 0, -1) : null, Poll.getOptionVoteCount(pollId, option), ]); + const options = tryParseOptions(optionsJson); + const optionData = options.find(opt => String(opt.id) === String(option)); + if (!optionData) { + return null; + } if (votes) { optionData.votes = votes; @@ -115,30 +149,28 @@ Poll.getOption = async function (pollId, option, withVotes = false) { }; Poll.getVotersCount = async function (pollId) { - return await NodeBB.db.sortedSetCard(`poll:${pollId}:voters`); + return await db.sortedSetCard(`poll:${pollId}:voters`); }; Poll.getVoteCount = async function (pollId) { - const options = await NodeBB.db.getSetMembers(`poll:${pollId}:options`); - return await NodeBB.db.sortedSetsCardSum( - options.map(option => `poll:${pollId}:options:${option}:votes`) + const optionIds = await Poll.getPollOptionIds(pollId); + return await db.sortedSetsCardSum( + optionIds.map(option => `poll:${pollId}:options:${option}:votes`) ); }; Poll.getOptionVoteCount = async function (pollId, option) { - return await NodeBB.db.sortedSetCard(`poll:${pollId}:options:${option}:votes`); + return await db.sortedSetCard(`poll:${pollId}:options:${option}:votes`); }; Poll.hasOption = async function (pollId, option) { - return await NodeBB.db.isSetMember(`poll:${pollId}:options`, option); + const optionIds = new Set(await Poll.getPollOptionIds(pollId)); + return optionIds.has(String(option)); }; Poll.hasOptions = async function (pollId, options) { - return await NodeBB.db.isSetMembers(`poll:${pollId}:options`, options); -}; - -Poll.getSettings = async function (pollId) { - return await NodeBB.db.getObject(`poll:${pollId}:settings`); + const optionIds = new Set(await Poll.getPollOptionIds(pollId)); + return options.map(option => optionIds.has(String(option))); }; Poll.isDeleted = async function (pollId) { @@ -146,60 +178,39 @@ Poll.isDeleted = async function (pollId) { return parseInt(result, 10) === 1; }; -Poll.delete = async function (pollId) { - await Poll.setField(pollId, 'deleted', 1); -}; - -Poll.restore = async function (pollId) { - await Poll.setFields(pollId, { - edited: 0, - deleted: 0, - }); -}; - -Poll.schedule = async function (pollId) { - await NodeBB.db.setAdd('polls:scheduled', pollId); - Scheduler.add(pollId); +Poll.delete = async function (pollIds) { + await db.setObject( + pollIds.map(pollId => `poll:${pollId}`), + { deleted: 1 }, + ); }; -Poll.getScheduled = async function () { - return await NodeBB.db.getSetMembers('polls:scheduled'); +Poll.restore = async function (pollIds) { + await db.setObject( + pollIds.map(pollId => `poll:${pollId}`), + { + edited: 0, + deleted: 0, + }, + ); }; Poll.hasEnded = async function (pollId) { - const result = await Poll.getField(pollId, 'ended'); - return parseInt(result, 10) === 1; + let end = await Poll.getField(pollId, 'end'); + end = parseInt(end, 10); + return end > 0 && Date.now() > end; }; Poll.doesDisallowVoteUpdate = async function (pollId) { - const result = await Poll.getSettingsField(pollId, 'disallowVoteUpdate'); + const result = await Poll.getField(pollId, 'disallowVoteUpdate'); return parseInt(result, 10) === 1; }; Poll.doesAllowVoteUpdate = async function (pollId) { - const result = await Poll.getSettingsField(pollId, 'disallowVoteUpdate'); + const result = await Poll.getField(pollId, 'disallowVoteUpdate'); return parseInt(result, 10) !== 1; }; -Poll.end = async function (pollId) { - await NodeBB.db.setRemove('polls:scheduled', pollId); - await Poll.setField(pollId, 'ended', 1); -}; - -Poll.changePid = async function (pollId, pid) { - await Promise.all([ - Poll.setField(pollId, 'pid', pid), - NodeBB.db.setObjectField(`post:${pid}`, 'pollId', pollId), - ]); -}; - -Poll.changeTid = async function (pollId, tid) { - await Promise.all([ - Poll.setField(pollId, 'tid', tid), - NodeBB.db.setObjectField(`topic:${tid}`, 'pollId', pollId), - ]); -}; - Poll.setField = async function (pollId, field, value) { await NodeBB.db.setObjectField(`poll:${pollId}`, field, value); }; @@ -216,11 +227,3 @@ Poll.getFields = async function (pollId, fields) { return await NodeBB.db.getObjectFields(`poll:${pollId}`, fields); }; -Poll.getSettingsField = async function (pollId, field) { - return await NodeBB.db.getObjectField(`poll:${pollId}:settings`, field); -}; - -Poll.setSettingsField = async function (pollId, field, value) { - await NodeBB.db.setObjectField(`poll:${pollId}:settings`, field, value); -}; - diff --git a/lib/scheduler.js b/lib/scheduler.js deleted file mode 100755 index 1578af4..0000000 --- a/lib/scheduler.js +++ /dev/null @@ -1,52 +0,0 @@ -'use strict'; - -const cron = require('cron').CronJob; -const NodeBB = require('./nodebb'); - -const Poll = require('./poll'); - -const Scheduler = exports; -const jobs = {}; - -Scheduler.start = async function () { - const pollIds = await Poll.getScheduled(); - await Promise.all(pollIds.map(Scheduler.add)); -}; - -Scheduler.add = async function (pollId) { - if (Object.keys(jobs).indexOf(pollId.toString()) !== -1) { - return; - } - - const settings = await Poll.getSettings(pollId); - if (!settings) { - return NodeBB.winston.error(`[nodebb-plugin-poll/scheduler] Poll ID ${pollId} has no settings!`); - } - - const now = Date.now(); - const end = parseInt(settings.end, 10); - - if (end < now) { - Scheduler.end(pollId); - } else { - const date = new Date(end); - NodeBB.winston.verbose(`[nodebb-plugin-poll/scheduler] Starting scheduler for poll with ID ${pollId} to end on ${date}`); - jobs[pollId] = new cron(date, (() => { - Scheduler.end(pollId); - }), null, true); - } -}; - -Scheduler.end = async function (pollId) { - NodeBB.winston.verbose(`[nodebb-plugin-poll/scheduler] Ending poll with ID ${pollId}`); - - const index = Object.keys(jobs).indexOf(pollId.toString()); - - if (index !== -1 && jobs[pollId] !== undefined) { - jobs[pollId].stop(); - delete jobs[pollId]; - } - - await Poll.end(pollId); -}; - diff --git a/lib/serializer.js b/lib/serializer.js deleted file mode 100755 index 29eac13..0000000 --- a/lib/serializer.js +++ /dev/null @@ -1,4 +0,0 @@ -'use strict'; - -const utils = require.main.require('./src/utils'); -module.exports = require('../public/js/poll/serializer')(utils); diff --git a/lib/sockets.js b/lib/sockets.js index 902168b..18e3a2c 100755 --- a/lib/sockets.js +++ b/lib/sockets.js @@ -1,43 +1,38 @@ 'use strict'; const _ = require.main.require('lodash'); -const NodeBB = require('./nodebb'); +const db = require.main.require('./src/database'); +const NodeBB = require('./nodebb'); const Config = require('./config'); const Poll = require('./poll'); const Vote = require('./vote'); const Sockets = exports; -Sockets.getConfig = function (socket, data, callback) { - callback(null, Config.settings.get()); -}; - Sockets.get = async function (socket, data) { - const allowAnon = Config.settings.get('toggles.allowAnon'); + const settings = await Config.getSettings(); + const allowAnon = parseInt(settings.allowGuestsToViewResults, 10) === 1; if (!data) { throw new Error('Invalid request, request data is not defined'); } - if (isNaN(parseInt(data.pollId, 10))) { + if (!Array.isArray(data.pollIds) || !data.pollIds.length) { throw new Error('Invalid request, pollId is required'); } if ((!socket.uid && !allowAnon)) { throw new Error('Invalid request, anonymous access is not allowed'); } + return await Promise.all(data.pollIds.map(async (pollId) => { + const pollData = await Poll.get(pollId, socket.uid, !!socket.uid); - const pollId = parseInt(data.pollId, 10); - const pollData = await Poll.get(pollId, socket.uid, !!socket.uid); - if (!pollData.info.version) { - throw new Error('Legacy polls are not supported'); - } - - if (parseInt(pollData.settings.allowAnonVoting, 10) === 1) { - await anonymizeVoters(socket.uid, data.pollId, pollData.options); - } + if (parseInt(pollData.allowAnonVoting, 10) === 1) { + await anonymizeVoters(socket.uid, pollId, pollData.options); + } - pollData.optionType = parseInt(pollData.settings.maxvotes, 10) > 1 ? 'checkbox' : 'radio'; - return pollData; + pollData.optionType = parseInt(pollData.maximumVotesPerUser, 10) > 1 ? 'checkbox' : 'radio'; + return pollData; + })); }; Sockets.vote = async function (socket, data) { @@ -50,21 +45,21 @@ Sockets.vote = async function (socket, data) { data.uid = socket.uid; - const [canVote, optionsFilter, settings] = await Promise.all([ + const [pollData, canVote, optionsFilter] = await Promise.all([ + db.getObject(`poll:${data.pollId}`), Vote.canVote(data.uid, data.pollId), Poll.hasOptions(data.pollId, data.options), - Poll.getSettings(data.pollId), ]); // Filter the options on their existence data.options = data.options.filter((el, index) => optionsFilter[index]); // Give an error if there are too many votes - if (data.options.length > parseInt(settings.maxvotes, 10)) { - throw new Error(`You can only vote for ${settings.maxvotes} options on this poll.`); + if (data.options.length > parseInt(pollData.maximumVotesPerUser, 10)) { + throw new Error(`You can only vote for ${pollData.maximumVotesPerUser} options on this poll.`); } - if (data.voteAnon && parseInt(settings.allowAnonVoting, 10) !== 1) { + if (data.voteAnon && parseInt(pollData.allowAnonVoting, 10) !== 1) { throw new Error('[[poll:error.anon-voting-not-allowed]]'); } @@ -74,8 +69,11 @@ Sockets.vote = async function (socket, data) { await Vote.add(data); - const pollData = await Poll.get(data.pollId, socket.uid, false); - NodeBB.SocketIndex.server.sockets.emit('event:poll.voteChange', { pollData, uid: socket.uid }); + const returnPoll = await Poll.get(data.pollId, socket.uid, false); + NodeBB.SocketIndex.server.sockets.emit('event:poll.voteChange', { + pollData: returnPoll, + uid: socket.uid, + }); }; Sockets.updateVote = async function (socket, data) { @@ -87,18 +85,18 @@ Sockets.updateVote = async function (socket, data) { } data.uid = socket.uid; - const [canUpdateVote, optionsFilter, settings] = await Promise.all([ + const [pollData, canUpdateVote, optionsFilter] = await Promise.all([ + db.getObject(`poll:${data.pollId}`), Vote.canUpdateVote(data.uid, data.pollId), Poll.hasOptions(data.pollId, data.options), - Poll.getSettings(data.pollId), ]); // Filter the options on their existence data.options = data.options.filter((el, index) => optionsFilter[index]); // Give an error if there are too many votes - if (data.options.length > parseInt(settings.maxvotes, 10)) { - throw new Error(`You can only vote for ${settings.maxvotes} options on this poll.`); + if (data.options.length > parseInt(pollData.maximumVotesPerUser, 10)) { + throw new Error(`You can only vote for ${pollData.maximumVotesPerUser} options on this poll.`); } if (!canUpdateVote) { @@ -110,8 +108,11 @@ Sockets.updateVote = async function (socket, data) { } await Vote.update(data); - const pollData = await Poll.get(data.pollId, socket.uid, false); - NodeBB.SocketIndex.server.sockets.emit('event:poll.voteChange', { pollData, uid: socket.uid }); + const returnPoll = await Poll.get(data.pollId, socket.uid, false); + NodeBB.SocketIndex.server.sockets.emit('event:poll.voteChange', { + pollData: returnPoll, + uid: socket.uid, + }); }; Sockets.removeVote = async function (socket, data) { @@ -130,24 +131,26 @@ Sockets.removeVote = async function (socket, data) { await Vote.remove(data); const pollData = await Poll.get(data.pollId, socket.uid, false); - NodeBB.SocketIndex.server.sockets.emit('event:poll.voteChange', { pollData, uid: socket.uid }); + NodeBB.SocketIndex.server.sockets.emit('event:poll.voteChange', { + pollData, + uid: socket.uid, + }); }; Sockets.getOptionDetails = async function (socket, data) { if (!socket.uid || !data || isNaN(parseInt(data.pollId, 10)) || isNaN(parseInt(data.optionId, 10))) { throw new Error('Invalid request'); } - const [poll, settings, option] = await Promise.all([ + const [poll, option] = await Promise.all([ Poll.getInfo(data.pollId), - Poll.getSettings(data.pollId), Poll.getOption(data.pollId, data.optionId, true), ]); - if (!option.votes || !option.votes.length) { + if (!option || !option.votes || !option.votes.length) { return option; } - if (parseInt(settings.allowAnonVoting, 10) === 1) { + if (parseInt(poll.allowAnonVoting, 10) === 1) { await anonymizeVoters(socket.uid, data.pollId, [option]); } @@ -164,15 +167,14 @@ Sockets.getOptionDetails = async function (socket, data) { }; Sockets.canCreate = async function (socket, data) { - if (!socket.uid || !data || (isNaN(parseInt(data.cid, 10)) && isNaN(parseInt(data.pid, 10)))) { + if (!socket.uid || !data) { throw new Error('Invalid request'); } - - if (!data.cid) { - const tid = await NodeBB.Posts.getPostField(data.pid, 'tid'); - data.cid = await NodeBB.Topics.getTopicField(tid, 'cid'); + let { cid } = data; + if (!cid) { + cid = await NodeBB.Topics.getTopicField(data.tid, 'cid'); } - return await checkPrivs(data.cid, socket.uid); + return await checkPrivs(cid, socket.uid); }; async function anonymizeVoters(callerUid, pollId, options) { diff --git a/lib/vote.js b/lib/vote.js index f167e34..7d24283 100755 --- a/lib/vote.js +++ b/lib/vote.js @@ -7,7 +7,7 @@ const Vote = exports; Vote.add = async function (voteData) { const { pollId, options, uid } = voteData; - const timestamp = +new Date(); + const timestamp = Date.now(); const promises = [ NodeBB.db.sortedSetAdd(`poll:${pollId}:voters`, timestamp, uid), diff --git a/library.js b/library.js index 12c22ba..4f8b6f7 100755 --- a/library.js +++ b/library.js @@ -1,12 +1,9 @@ 'use strict'; -const nconf = require.main.require('nconf'); - const NodeBB = require('./lib/nodebb'); const Config = require('./lib/config'); const Sockets = require('./lib/sockets'); const Hooks = require('./lib/hooks'); -const Scheduler = require('./lib/scheduler'); const Plugin = module.exports; @@ -23,15 +20,8 @@ Plugin.load = async function (params) { }); NodeBB.PluginSockets[Config.plugin.id] = Sockets; - NodeBB.AdminSockets[Config.plugin.id] = Config.adminSockets; NodeBB.app = params.app; - - if (nconf.get('runJobs')) { - Scheduler.start(); - } - - await Config.init(); }; Plugin.addAdminNavigation = function (adminHeader) { @@ -48,6 +38,7 @@ Plugin.registerFormatting = function (payload) { name: 'poll', className: `fa ${Config.plugin.icon}`, title: '[[poll:creator_title]]', + badge: true, }); return payload; }; @@ -82,12 +73,8 @@ Plugin.defineWidgets = async function (widgets) { }; Plugin.renderPollWidget = async function (widget) { - const { tid } = widget.data; + const { pollId } = widget.data; const Poll = require('./lib/poll'); - const pollId = await Poll.getPollIdByTid(tid); - if (!pollId) { - return null; - } const pollData = await Poll.get(pollId, widget.uid, widget.req.loggedIn); if (!pollData) { return null; diff --git a/package-lock.json b/package-lock.json index 8a5c238..2f3c225 100644 --- a/package-lock.json +++ b/package-lock.json @@ -13,14 +13,14 @@ "cron": "^1.8.2" }, "devDependencies": { - "eslint": "9.28.0", - "eslint-config-nodebb": "1.1.7" + "eslint": "10.0.3", + "eslint-config-nodebb": "^2.0.0" } }, "node_modules/@eslint-community/eslint-utils": { - "version": "4.7.0", - "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.7.0.tgz", - "integrity": "sha512-dyybb3AcajC7uha6CvhdVRJqaKyn7w2YKqKyAN37NKYgZT36w+iRb0Dymmc5qEJ549c/S31cMMSFd75bteCpCw==", + "version": "4.9.1", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.1.tgz", + "integrity": "sha512-phrYmNiYppR7znFEdqgfWHXR6NCkZEK7hwWDHZUjit/2/U0r6XvkDl0SYnoM51Hq7FhCGdLDT6zxCCOY1hexsQ==", "dev": true, "dependencies": { "eslint-visitor-keys": "^3.4.3" @@ -36,104 +36,93 @@ } }, "node_modules/@eslint-community/regexpp": { - "version": "4.12.1", - "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.1.tgz", - "integrity": "sha512-CCZCDJuduB9OUkFkY2IgppNZMi2lBQgD2qzwXkEia16cge2pijY/aXi96CJMquDMn3nJdlPV1A5KrJEXwfLNzQ==", + "version": "4.12.2", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz", + "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==", "dev": true, "engines": { "node": "^12.0.0 || ^14.0.0 || >=16.0.0" } }, "node_modules/@eslint/config-array": { - "version": "0.20.0", - "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.20.0.tgz", - "integrity": "sha512-fxlS1kkIjx8+vy2SjuCB94q3htSNrufYTXubwiBFeaQHbH6Ipi43gFJq2zCMt6PHhImH3Xmr0NksKDvchWlpQQ==", + "version": "0.23.3", + "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.23.3.tgz", + "integrity": "sha512-j+eEWmB6YYLwcNOdlwQ6L2OsptI/LO6lNBuLIqe5R7RetD658HLoF+Mn7LzYmAWWNNzdC6cqP+L6r8ujeYXWLw==", "dev": true, "dependencies": { - "@eslint/object-schema": "^2.1.6", + "@eslint/object-schema": "^3.0.3", "debug": "^4.3.1", - "minimatch": "^3.1.2" + "minimatch": "^10.2.4" }, "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + "node": "^20.19.0 || ^22.13.0 || >=24" } }, "node_modules/@eslint/config-helpers": { - "version": "0.2.2", - "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.2.2.tgz", - "integrity": "sha512-+GPzk8PlG0sPpzdU5ZvIRMPidzAnZDl/s9L+y13iodqvb8leL53bTannOrQ/Im7UkpsmFU5Ily5U60LWixnmLg==", - "dev": true, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - } - }, - "node_modules/@eslint/core": { - "version": "0.14.0", - "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.14.0.tgz", - "integrity": "sha512-qIbV0/JZr7iSDjqAc60IqbLdsj9GDt16xQtWD+B78d/HAlvysGdZZ6rpJHGAc2T0FQx1X6thsSPdnoiGKdNtdg==", + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.5.3.tgz", + "integrity": "sha512-lzGN0onllOZCGroKJmRwY6QcEHxbjBw1gwB8SgRSqK8YbbtEXMvKynsXc3553ckIEBxsbMBU7oOZXKIPGZNeZw==", "dev": true, "dependencies": { - "@types/json-schema": "^7.0.15" + "@eslint/core": "^1.1.1" }, "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + "node": "^20.19.0 || ^22.13.0 || >=24" } }, - "node_modules/@eslint/eslintrc": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.3.1.tgz", - "integrity": "sha512-gtF186CXhIl1p4pJNGZw8Yc6RlshoePRvE0X91oPGb3vZ8pM3qOS9W9NGPat9LziaBV7XrJWGylNQXkGcnM3IQ==", + "node_modules/@eslint/core": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@eslint/core/-/core-1.1.1.tgz", + "integrity": "sha512-QUPblTtE51/7/Zhfv8BDwO0qkkzQL7P/aWWbqcf4xWLEYn1oKjdO0gglQBB4GAsu7u6wjijbCmzsUTy6mnk6oQ==", "dev": true, "dependencies": { - "ajv": "^6.12.4", - "debug": "^4.3.2", - "espree": "^10.0.1", - "globals": "^14.0.0", - "ignore": "^5.2.0", - "import-fresh": "^3.2.1", - "js-yaml": "^4.1.0", - "minimatch": "^3.1.2", - "strip-json-comments": "^3.1.1" + "@types/json-schema": "^7.0.15" }, "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" + "node": "^20.19.0 || ^22.13.0 || >=24" } }, "node_modules/@eslint/js": { - "version": "9.28.0", - "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.28.0.tgz", - "integrity": "sha512-fnqSjGWd/CoIp4EXIxWVK/sHA6DOHN4+8Ix2cX5ycOY7LG0UY8nHCU5pIp2eaE1Mc7Qd8kHspYNzYXT2ojPLzg==", + "version": "10.0.1", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-10.0.1.tgz", + "integrity": "sha512-zeR9k5pd4gxjZ0abRoIaxdc7I3nDktoXZk2qOv9gCNWx3mVwEn32VRhyLaRsDiJjTs0xq/T8mfPtyuXu7GWBcA==", "dev": true, + "peer": true, "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + "node": "^20.19.0 || ^22.13.0 || >=24" }, "funding": { "url": "https://eslint.org/donate" + }, + "peerDependencies": { + "eslint": "^10.0.0" + }, + "peerDependenciesMeta": { + "eslint": { + "optional": true + } } }, "node_modules/@eslint/object-schema": { - "version": "2.1.6", - "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.6.tgz", - "integrity": "sha512-RBMg5FRL0I0gs51M/guSAj5/e14VQ4tpZnQNWwuDT66P14I43ItmPfIZRhO9fUVIPOAQXU47atlywZ/czoqFPA==", + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-3.0.3.tgz", + "integrity": "sha512-iM869Pugn9Nsxbh/YHRqYiqd23AmIbxJOcpUMOuWCVNdoQJ5ZtwL6h3t0bcZzJUlC3Dq9jCFCESBZnX0GTv7iQ==", "dev": true, "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + "node": "^20.19.0 || ^22.13.0 || >=24" } }, "node_modules/@eslint/plugin-kit": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.3.1.tgz", - "integrity": "sha512-0J+zgWxHN+xXONWIyPWKFMgVuJoZuGiIFu8yxk7RJjxkzpGmyja5wRFqZIVtjDVOQpV+Rw0iOAjYPE2eQyjr0w==", + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.6.1.tgz", + "integrity": "sha512-iH1B076HoAshH1mLpHMgwdGeTs0CYwL0SPMkGuSebZrwBp16v415e9NZXg2jtrqPVQjf6IANe2Vtlr5KswtcZQ==", "dev": true, "dependencies": { - "@eslint/core": "^0.14.0", + "@eslint/core": "^1.1.1", "levn": "^0.4.1" }, "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + "node": "^20.19.0 || ^22.13.0 || >=24" } }, "node_modules/@humanfs/core": { @@ -197,69 +186,25 @@ "url": "https://github.com/sponsors/nzakas" } }, - "node_modules/@nodelib/fs.scandir": { - "version": "2.1.5", - "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", - "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", - "dev": true, - "peer": true, - "dependencies": { - "@nodelib/fs.stat": "2.0.5", - "run-parallel": "^1.1.9" - }, - "engines": { - "node": ">= 8" - } - }, - "node_modules/@nodelib/fs.stat": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", - "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 8" - } - }, - "node_modules/@nodelib/fs.walk": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", - "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", - "dev": true, - "peer": true, - "dependencies": { - "@nodelib/fs.scandir": "2.1.5", - "fastq": "^1.6.0" - }, - "engines": { - "node": ">= 8" - } - }, - "node_modules/@rtsao/scc": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@rtsao/scc/-/scc-1.1.0.tgz", - "integrity": "sha512-zt6OdqaDoOnJ1ZYsCYGt9YmWzDXl4vQdKTyJev62gFhRGKdx7mcT54V9KIjg+d2wi9EXsPvAPKe7i7WjfVWB8g==", - "dev": true, - "peer": true - }, "node_modules/@stylistic/eslint-plugin": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/@stylistic/eslint-plugin/-/eslint-plugin-4.4.1.tgz", - "integrity": "sha512-CEigAk7eOLyHvdgmpZsKFwtiqS2wFwI1fn4j09IU9GmD4euFM4jEBAViWeCqaNLlbX2k2+A/Fq9cje4HQBXuJQ==", + "version": "5.10.0", + "resolved": "https://registry.npmjs.org/@stylistic/eslint-plugin/-/eslint-plugin-5.10.0.tgz", + "integrity": "sha512-nPK52ZHvot8Ju/0A4ucSX1dcPV2/1clx0kLcH5wDmrE4naKso7TUC/voUyU1O9OTKTrR6MYip6LP0ogEMQ9jPQ==", "dev": true, "peer": true, "dependencies": { - "@typescript-eslint/utils": "^8.32.1", - "eslint-visitor-keys": "^4.2.0", - "espree": "^10.3.0", + "@eslint-community/eslint-utils": "^4.9.1", + "@typescript-eslint/types": "^8.56.0", + "eslint-visitor-keys": "^4.2.1", + "espree": "^10.4.0", "estraverse": "^5.3.0", - "picomatch": "^4.0.2" + "picomatch": "^4.0.3" }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "peerDependencies": { - "eslint": ">=9.0.0" + "eslint": "^9.0.0 || ^10.0.0" } }, "node_modules/@stylistic/eslint-plugin/node_modules/eslint-visitor-keys": { @@ -275,6 +220,12 @@ "url": "https://opencollective.com/eslint" } }, + "node_modules/@types/esrecurse": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/@types/esrecurse/-/esrecurse-4.3.1.tgz", + "integrity": "sha512-xJBAbDifo5hpffDBuHl0Y8ywswbiAp/Wi7Y/GtAgSlZyIABppyurxVueOPE8LUQOxdlgi6Zqce7uoEpqNTeiUw==", + "dev": true + }, "node_modules/@types/estree": { "version": "1.0.8", "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz", @@ -287,182 +238,286 @@ "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", "dev": true }, - "node_modules/@types/json5": { - "version": "0.0.29", - "resolved": "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz", - "integrity": "sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==", - "dev": true, - "peer": true - }, - "node_modules/@typescript-eslint/project-service": { - "version": "8.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/project-service/-/project-service-8.34.0.tgz", - "integrity": "sha512-iEgDALRf970/B2YExmtPMPF54NenZUf4xpL3wsCRx/lgjz6ul/l13R81ozP/ZNuXfnLCS+oPmG7JIxfdNYKELw==", + "node_modules/@typescript-eslint/types": { + "version": "8.56.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.56.1.tgz", + "integrity": "sha512-dbMkdIUkIkchgGDIv7KLUpa0Mda4IYjo4IAMJUZ+3xNoUXxMsk9YtKpTHSChRS85o+H9ftm51gsK1dZReY9CVw==", "dev": true, "peer": true, - "dependencies": { - "@typescript-eslint/tsconfig-utils": "^8.34.0", - "@typescript-eslint/types": "^8.34.0", - "debug": "^4.3.4" - }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { "type": "opencollective", "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/acorn": { + "version": "8.16.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.16.0.tgz", + "integrity": "sha512-UVJyE9MttOsBQIDKw1skb9nAwQuR5wuGD3+82K6JgJlm/Y+KI92oNsMNGZCYdDsVtRHSak0pcV5Dno5+4jh9sw==", + "dev": true, + "bin": { + "acorn": "bin/acorn" }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, "peerDependencies": { - "typescript": ">=4.8.4 <5.9.0" + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" } }, - "node_modules/@typescript-eslint/scope-manager": { - "version": "8.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.34.0.tgz", - "integrity": "sha512-9Ac0X8WiLykl0aj1oYQNcLZjHgBojT6cW68yAgZ19letYu+Hxd0rE0veI1XznSSst1X5lwnxhPbVdwjDRIomRw==", + "node_modules/ajv": { + "version": "6.14.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.14.0.tgz", + "integrity": "sha512-IWrosm/yrn43eiKqkfkHis7QioDleaXQHdDVPKg0FSwwd/DuvyX79TZnFOnYpB7dcsFAMmtFztZuXPDvSePkFw==", "dev": true, - "peer": true, "dependencies": { - "@typescript-eslint/types": "8.34.0", - "@typescript-eslint/visitor-keys": "8.34.0" + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/async": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/async/-/async-3.2.0.tgz", + "integrity": "sha512-TR2mEZFVOj2pLStYxLht7TyfuRzaydfpxr3k9RpHIzMgw7A64dzsdqCxH1WJyQdoe8T10nDXd9wnEigmiuHIZw==" + }, + "node_modules/balanced-match": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-4.0.4.tgz", + "integrity": "sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA==", + "dev": true, "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + "node": "18 || 20 || >=22" + } + }, + "node_modules/brace-expansion": { + "version": "5.0.4", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-5.0.4.tgz", + "integrity": "sha512-h+DEnpVvxmfVefa4jFbCf5HdH5YMDXRsmKflpf1pILZWRFlTbJpxeU55nJl4Smt5HQaGzg1o6RHFPJaOqnmBDg==", + "dev": true, + "dependencies": { + "balanced-match": "^4.0.2" }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" + "engines": { + "node": "18 || 20 || >=22" + } + }, + "node_modules/cron": { + "version": "1.8.2", + "resolved": "https://registry.npmjs.org/cron/-/cron-1.8.2.tgz", + "integrity": "sha512-Gk2c4y6xKEO8FSAUTklqtfSr7oTq0CiPQeLBG5Fl0qoXpZyMcj1SG59YL+hqq04bu6/IuEA7lMkYDAplQNKkyg==", + "dependencies": { + "moment-timezone": "^0.5.x" } }, - "node_modules/@typescript-eslint/tsconfig-utils": { - "version": "8.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/tsconfig-utils/-/tsconfig-utils-8.34.0.tgz", - "integrity": "sha512-+W9VYHKFIzA5cBeooqQxqNriAP0QeQ7xTiDuIOr71hzgffm3EL2hxwWBIIj4GuofIbKxGNarpKqIq6Q6YrShOA==", + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", "dev": true, - "peer": true, + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + "node": ">= 8" + } + }, + "node_modules/debug": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.1.tgz", + "integrity": "sha512-KcKCqiftBJcZr++7ykoDIEwSa3XWowTfNPo92BYxjXiyYEVrUQh2aLyhxBCwww+heortUFxEJYcRzosstTEBYQ==", + "dev": true, + "dependencies": { + "ms": "^2.1.3" }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" + "engines": { + "node": ">=6.0" }, - "peerDependencies": { - "typescript": ">=4.8.4 <5.9.0" + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } } }, - "node_modules/@typescript-eslint/types": { - "version": "8.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.34.0.tgz", - "integrity": "sha512-9V24k/paICYPniajHfJ4cuAWETnt7Ssy+R0Rbcqo5sSFr3QEZ/8TSoUi9XeXVBGXCaLtwTOKSLGcInCAvyZeMA==", + "node_modules/deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true + }, + "node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", "dev": true, - "peer": true, "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + "node": ">=10" }, "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" + "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/@typescript-eslint/typescript-estree": { - "version": "8.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.34.0.tgz", - "integrity": "sha512-rOi4KZxI7E0+BMqG7emPSK1bB4RICCpF7QD3KCLXn9ZvWoESsOMlHyZPAHyG04ujVplPaHbmEvs34m+wjgtVtg==", + "node_modules/eslint": { + "version": "10.0.3", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-10.0.3.tgz", + "integrity": "sha512-COV33RzXZkqhG9P2rZCFl9ZmJ7WL+gQSCRzE7RhkbclbQPtLAWReL7ysA0Sh4c8Im2U9ynybdR56PV0XcKvqaQ==", "dev": true, - "peer": true, "dependencies": { - "@typescript-eslint/project-service": "8.34.0", - "@typescript-eslint/tsconfig-utils": "8.34.0", - "@typescript-eslint/types": "8.34.0", - "@typescript-eslint/visitor-keys": "8.34.0", - "debug": "^4.3.4", - "fast-glob": "^3.3.2", - "is-glob": "^4.0.3", - "minimatch": "^9.0.4", - "semver": "^7.6.0", - "ts-api-utils": "^2.1.0" + "@eslint-community/eslint-utils": "^4.8.0", + "@eslint-community/regexpp": "^4.12.2", + "@eslint/config-array": "^0.23.3", + "@eslint/config-helpers": "^0.5.2", + "@eslint/core": "^1.1.1", + "@eslint/plugin-kit": "^0.6.1", + "@humanfs/node": "^0.16.6", + "@humanwhocodes/module-importer": "^1.0.1", + "@humanwhocodes/retry": "^0.4.2", + "@types/estree": "^1.0.6", + "ajv": "^6.14.0", + "cross-spawn": "^7.0.6", + "debug": "^4.3.2", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^9.1.2", + "eslint-visitor-keys": "^5.0.1", + "espree": "^11.1.1", + "esquery": "^1.7.0", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^8.0.0", + "find-up": "^5.0.0", + "glob-parent": "^6.0.2", + "ignore": "^5.2.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "minimatch": "^10.2.4", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3" + }, + "bin": { + "eslint": "bin/eslint.js" }, "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + "node": "^20.19.0 || ^22.13.0 || >=24" }, "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" + "url": "https://eslint.org/donate" }, "peerDependencies": { - "typescript": ">=4.8.4 <5.9.0" + "jiti": "*" + }, + "peerDependenciesMeta": { + "jiti": { + "optional": true + } } }, - "node_modules/@typescript-eslint/typescript-estree/node_modules/brace-expansion": { + "node_modules/eslint-config-nodebb": { "version": "2.0.1", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", - "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", + "resolved": "https://registry.npmjs.org/eslint-config-nodebb/-/eslint-config-nodebb-2.0.1.tgz", + "integrity": "sha512-06QoLSEtjcUOstURvS4T2GyTFQOUdD/dCvJkrNi+CQ+9uJPpM41uZQAKJi94gv9OFTeLSdoqxESUXLbw2czqjw==", "dev": true, - "peer": true, "dependencies": { - "balanced-match": "^1.0.0" + "globals": "17.3.0" + }, + "peerDependencies": { + "@eslint/js": "^10.0.0", + "@stylistic/eslint-plugin": "^5.x" } }, - "node_modules/@typescript-eslint/typescript-estree/node_modules/minimatch": { - "version": "9.0.5", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", - "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", + "node_modules/eslint-scope": { + "version": "9.1.2", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-9.1.2.tgz", + "integrity": "sha512-xS90H51cKw0jltxmvmHy2Iai1LIqrfbw57b79w/J7MfvDfkIkFZ+kj6zC3BjtUwh150HsSSdxXZcsuv72miDFQ==", "dev": true, - "peer": true, "dependencies": { - "brace-expansion": "^2.0.1" + "@types/esrecurse": "^4.3.1", + "@types/estree": "^1.0.8", + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" }, "engines": { - "node": ">=16 || 14 >=14.17" + "node": "^20.19.0 || ^22.13.0 || >=24" }, "funding": { - "url": "https://github.com/sponsors/isaacs" + "url": "https://opencollective.com/eslint" } }, - "node_modules/@typescript-eslint/utils": { - "version": "8.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.34.0.tgz", - "integrity": "sha512-8L4tWatGchV9A1cKbjaavS6mwYwp39jql8xUmIIKJdm+qiaeHy5KMKlBrf30akXAWBzn2SqKsNOtSENWUwg7XQ==", + "node_modules/eslint-visitor-keys": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", + "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", "dev": true, - "peer": true, - "dependencies": { - "@eslint-community/eslint-utils": "^4.7.0", - "@typescript-eslint/scope-manager": "8.34.0", - "@typescript-eslint/types": "8.34.0", - "@typescript-eslint/typescript-estree": "8.34.0" + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint/node_modules/eslint-visitor-keys": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-5.0.1.tgz", + "integrity": "sha512-tD40eHxA35h0PEIZNeIjkHoDR4YjjJp34biM0mDvplBe//mB+IHCqHDGV7pxF+7MklTvighcCPPZC7ynWyjdTA==", + "dev": true, "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + "node": "^20.19.0 || ^22.13.0 || >=24" }, "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint/node_modules/espree": { + "version": "11.2.0", + "resolved": "https://registry.npmjs.org/espree/-/espree-11.2.0.tgz", + "integrity": "sha512-7p3DrVEIopW1B1avAGLuCSh1jubc01H2JHc8B4qqGblmg5gI9yumBgACjWo4JlIc04ufug4xJ3SQI8HkS/Rgzw==", + "dev": true, + "dependencies": { + "acorn": "^8.16.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^5.0.1" }, - "peerDependencies": { - "eslint": "^8.57.0 || ^9.0.0", - "typescript": ">=4.8.4 <5.9.0" + "engines": { + "node": "^20.19.0 || ^22.13.0 || >=24" + }, + "funding": { + "url": "https://opencollective.com/eslint" } }, - "node_modules/@typescript-eslint/visitor-keys": { - "version": "8.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.34.0.tgz", - "integrity": "sha512-qHV7pW7E85A0x6qyrFn+O+q1k1p3tQCsqIZ1KZ5ESLXY57aTvUd3/a4rdPTeXisvhXn2VQG0VSKUqs8KHF2zcA==", + "node_modules/espree": { + "version": "10.4.0", + "resolved": "https://registry.npmjs.org/espree/-/espree-10.4.0.tgz", + "integrity": "sha512-j6PAQ2uUr79PZhBjP5C5fhl8e39FmRnOjsD5lGnWrFU8i2G776tBK7+nP8KuQUTTyAZUwfQqXAgrVH5MbH9CYQ==", "dev": true, "peer": true, "dependencies": { - "@typescript-eslint/types": "8.34.0", - "eslint-visitor-keys": "^4.2.0" + "acorn": "^8.15.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^4.2.1" }, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" }, "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" + "url": "https://opencollective.com/eslint" } }, - "node_modules/@typescript-eslint/visitor-keys/node_modules/eslint-visitor-keys": { + "node_modules/espree/node_modules/eslint-visitor-keys": { "version": "4.2.1", "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.1.tgz", "integrity": "sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ==", @@ -475,2835 +530,410 @@ "url": "https://opencollective.com/eslint" } }, - "node_modules/acorn": { - "version": "8.15.0", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz", - "integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==", + "node_modules/esquery": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.7.0.tgz", + "integrity": "sha512-Ap6G0WQwcU/LHsvLwON1fAQX9Zp0A2Y6Y/cJBl9r/JbW90Zyg4/zbG6zzKa2OTALELarYHmKu0GhpM5EO+7T0g==", "dev": true, - "bin": { - "acorn": "bin/acorn" + "dependencies": { + "estraverse": "^5.1.0" }, "engines": { - "node": ">=0.4.0" + "node": ">=0.10" } }, - "node_modules/acorn-jsx": { - "version": "5.3.2", - "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", - "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "node_modules/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", "dev": true, - "peerDependencies": { - "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" } }, - "node_modules/ajv": { - "version": "6.12.6", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", - "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", "dev": true, - "dependencies": { - "fast-deep-equal": "^3.1.1", - "fast-json-stable-stringify": "^2.0.0", - "json-schema-traverse": "^0.4.1", - "uri-js": "^4.2.2" - }, - "funding": { - "type": "github", - "url": "https://github.com/sponsors/epoberezkin" + "engines": { + "node": ">=4.0" } }, - "node_modules/argparse": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", - "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true + }, + "node_modules/fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", "dev": true }, - "node_modules/array-buffer-byte-length": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/array-buffer-byte-length/-/array-buffer-byte-length-1.0.2.tgz", - "integrity": "sha512-LHE+8BuR7RYGDKvnrmcuSq3tDcKv9OFEXQt/HpbZhY7V6h0zlUXutnAD82GiFx9rdieCMjkvtcsPqBwgUl1Iiw==", + "node_modules/file-entry-cache": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz", + "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==", "dev": true, - "peer": true, "dependencies": { - "call-bound": "^1.0.3", - "is-array-buffer": "^3.0.5" + "flat-cache": "^4.0.0" }, "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "node": ">=16.0.0" } }, - "node_modules/array-includes": { - "version": "3.1.9", - "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.9.tgz", - "integrity": "sha512-FmeCCAenzH0KH381SPT5FZmiA/TmpndpcaShhfgEN9eCVjnFBqq3l1xrI42y8+PPLI6hypzou4GXw00WHmPBLQ==", + "node_modules/find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", "dev": true, - "peer": true, "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.4", - "define-properties": "^1.2.1", - "es-abstract": "^1.24.0", - "es-object-atoms": "^1.1.1", - "get-intrinsic": "^1.3.0", - "is-string": "^1.1.1", - "math-intrinsics": "^1.1.0" + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" }, "engines": { - "node": ">= 0.4" + "node": ">=10" }, "funding": { - "url": "https://github.com/sponsors/ljharb" + "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/array.prototype.findlastindex": { - "version": "1.2.6", - "resolved": "https://registry.npmjs.org/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.6.tgz", - "integrity": "sha512-F/TKATkzseUExPlfvmwQKGITM3DGTK+vkAsCZoDc5daVygbJBnjEUCbgkAvVFsgfXfX4YIqZ/27G3k3tdXrTxQ==", + "node_modules/flat-cache": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz", + "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==", "dev": true, - "peer": true, "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.4", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.9", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.1.1", - "es-shim-unscopables": "^1.1.0" + "flatted": "^3.2.9", + "keyv": "^4.5.4" }, "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "node": ">=16" } }, - "node_modules/array.prototype.flat": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.3.3.tgz", - "integrity": "sha512-rwG/ja1neyLqCuGZ5YYrznA62D4mZXg0i1cIskIUKSiqF3Cje9/wXAls9B9s1Wa2fomMsIv8czB8jZcPmxCXFg==", + "node_modules/flatted": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.3.tgz", + "integrity": "sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg==", + "dev": true + }, + "node_modules/glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", "dev": true, - "peer": true, "dependencies": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-shim-unscopables": "^1.0.2" + "is-glob": "^4.0.3" }, "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "node": ">=10.13.0" } }, - "node_modules/array.prototype.flatmap": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/array.prototype.flatmap/-/array.prototype.flatmap-1.3.3.tgz", - "integrity": "sha512-Y7Wt51eKJSyi80hFrJCePGGNo5ktJCslFuboqJsbf57CCPcm5zztluPlc4/aD8sWsKvlwatezpV4U1efk8kpjg==", + "node_modules/globals": { + "version": "17.3.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-17.3.0.tgz", + "integrity": "sha512-yMqGUQVVCkD4tqjOJf3TnrvaaHDMYp4VlUSObbkIiuCPe/ofdMBFIAcBbCSRFWOnos6qRiTVStDwqPLUclaxIw==", "dev": true, - "peer": true, - "dependencies": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-shim-unscopables": "^1.0.2" - }, "engines": { - "node": ">= 0.4" + "node": ">=18" }, "funding": { - "url": "https://github.com/sponsors/ljharb" + "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/arraybuffer.prototype.slice": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.4.tgz", - "integrity": "sha512-BNoCY6SXXPQ7gF2opIP4GBE+Xw7U+pHMYKuzjgCN3GwiaIR09UUeKfheyIry77QtrCBlC0KK0q5/TER/tYh3PQ==", + "node_modules/ignore": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", "dev": true, - "peer": true, - "dependencies": { - "array-buffer-byte-length": "^1.0.1", - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6", - "is-array-buffer": "^3.0.4" - }, "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "node": ">= 4" } }, - "node_modules/async": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/async/-/async-3.2.0.tgz", - "integrity": "sha512-TR2mEZFVOj2pLStYxLht7TyfuRzaydfpxr3k9RpHIzMgw7A64dzsdqCxH1WJyQdoe8T10nDXd9wnEigmiuHIZw==" + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", + "dev": true, + "engines": { + "node": ">=0.8.19" + } }, - "node_modules/async-function": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/async-function/-/async-function-1.0.0.tgz", - "integrity": "sha512-hsU18Ae8CDTR6Kgu9DYf0EbCr/a5iGL0rytQDobUcdpYOKokk8LEjVphnXkDkgpi0wYVsqrXuP0bZxJaTqdgoA==", + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", "dev": true, - "peer": true, "engines": { - "node": ">= 0.4" + "node": ">=0.10.0" } }, - "node_modules/available-typed-arrays": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", - "integrity": "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==", + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", "dev": true, - "peer": true, "dependencies": { - "possible-typed-array-names": "^1.0.0" + "is-extglob": "^2.1.1" }, "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "node": ">=0.10.0" } }, - "node_modules/balanced-match": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", - "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", "dev": true }, - "node_modules/brace-expansion": { - "version": "1.1.11", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", - "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "node_modules/json-buffer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", + "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", + "dev": true + }, + "node_modules/json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true + }, + "node_modules/json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", + "dev": true + }, + "node_modules/keyv": { + "version": "4.5.4", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", + "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", "dev": true, "dependencies": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" + "json-buffer": "3.0.1" } }, - "node_modules/braces": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", - "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", + "node_modules/levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", "dev": true, - "peer": true, "dependencies": { - "fill-range": "^7.1.1" + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" }, "engines": { - "node": ">=8" + "node": ">= 0.8.0" } }, - "node_modules/call-bind": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.8.tgz", - "integrity": "sha512-oKlSFMcMwpUg2ednkhQ454wfWiU/ul3CkJe/PEHcTKuiX6RpbehUiFMXu13HalGZxfUwCQzZG747YXBn1im9ww==", + "node_modules/locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", "dev": true, - "peer": true, "dependencies": { - "call-bind-apply-helpers": "^1.0.0", - "es-define-property": "^1.0.0", - "get-intrinsic": "^1.2.4", - "set-function-length": "^1.2.2" + "p-locate": "^5.0.0" }, "engines": { - "node": ">= 0.4" + "node": ">=10" }, "funding": { - "url": "https://github.com/sponsors/ljharb" + "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/call-bind-apply-helpers": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", - "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", + "node_modules/minimatch": { + "version": "10.2.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.4.tgz", + "integrity": "sha512-oRjTw/97aTBN0RHbYCdtF1MQfvusSIBQM0IZEgzl6426+8jSC0nF1a/GmnVLpfB9yyr6g6FTqWqiZVbxrtaCIg==", "dev": true, - "peer": true, "dependencies": { - "es-errors": "^1.3.0", - "function-bind": "^1.1.2" + "brace-expansion": "^5.0.2" }, "engines": { - "node": ">= 0.4" + "node": "18 || 20 || >=22" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" } }, - "node_modules/call-bound": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz", - "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==", - "dev": true, - "peer": true, + "node_modules/moment": { + "version": "2.29.1", + "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.1.tgz", + "integrity": "sha512-kHmoybcPV8Sqy59DwNDY3Jefr64lK/by/da0ViFcuA4DH0vQg5Q6Ze5VimxkfQNSC+Mls/Kx53s7TjP1RhFEDQ==", + "engines": { + "node": "*" + } + }, + "node_modules/moment-timezone": { + "version": "0.5.32", + "resolved": "https://registry.npmjs.org/moment-timezone/-/moment-timezone-0.5.32.tgz", + "integrity": "sha512-Z8QNyuQHQAmWucp8Knmgei8YNo28aLjJq6Ma+jy1ZSpSk5nyfRT8xgUbSQvD2+2UajISfenndwvFuH3NGS+nvA==", "dependencies": { - "call-bind-apply-helpers": "^1.0.2", - "get-intrinsic": "^1.3.0" + "moment": ">= 2.9.0" }, "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "node": "*" } }, - "node_modules/callsites": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", - "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", + "dev": true + }, + "node_modules/optionator": { + "version": "0.9.4", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", + "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", "dev": true, + "dependencies": { + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0", + "word-wrap": "^1.2.5" + }, "engines": { - "node": ">=6" + "node": ">= 0.8.0" } }, - "node_modules/chalk": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.1.tgz", - "integrity": "sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==", + "node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", "dev": true, "dependencies": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "yocto-queue": "^0.1.0" }, "engines": { "node": ">=10" }, "funding": { - "url": "https://github.com/chalk/chalk?sponsor=1" + "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/chalk/node_modules/ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "node_modules/p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", "dev": true, "dependencies": { - "color-convert": "^2.0.1" + "p-limit": "^3.0.2" }, "engines": { - "node": ">=8" + "node": ">=10" }, "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" + "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/chalk/node_modules/color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", - "dev": true, - "dependencies": { - "color-name": "~1.1.4" - }, - "engines": { - "node": ">=7.0.0" - } - }, - "node_modules/chalk/node_modules/color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true - }, - "node_modules/chalk/node_modules/has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true, - "engines": { - "node": ">=8" - } - }, - "node_modules/chalk/node_modules/supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", - "dev": true, - "dependencies": { - "has-flag": "^4.0.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/concat-map": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", - "dev": true - }, - "node_modules/cron": { - "version": "1.8.2", - "resolved": "https://registry.npmjs.org/cron/-/cron-1.8.2.tgz", - "integrity": "sha512-Gk2c4y6xKEO8FSAUTklqtfSr7oTq0CiPQeLBG5Fl0qoXpZyMcj1SG59YL+hqq04bu6/IuEA7lMkYDAplQNKkyg==", - "dependencies": { - "moment-timezone": "^0.5.x" - } - }, - "node_modules/cross-spawn": { - "version": "7.0.6", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", - "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", - "dev": true, - "dependencies": { - "path-key": "^3.1.0", - "shebang-command": "^2.0.0", - "which": "^2.0.1" - }, - "engines": { - "node": ">= 8" - } - }, - "node_modules/data-view-buffer": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/data-view-buffer/-/data-view-buffer-1.0.2.tgz", - "integrity": "sha512-EmKO5V3OLXh1rtK2wgXRansaK1/mtVdTUEiEI0W8RkvgT05kfxaH29PliLnpLP73yYO6142Q72QNa8Wx/A5CqQ==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/data-view-byte-length": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/data-view-byte-length/-/data-view-byte-length-1.0.2.tgz", - "integrity": "sha512-tuhGbE6CfTM9+5ANGf+oQb72Ky/0+s3xKUpHvShfiz2RxMFgFPjsXuRLBVMtvMs15awe45SRb83D6wH4ew6wlQ==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/inspect-js" - } - }, - "node_modules/data-view-byte-offset": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/data-view-byte-offset/-/data-view-byte-offset-1.0.1.tgz", - "integrity": "sha512-BS8PfmtDGnrgYdOonGZQdLZslWIeCGFP9tpan0hi1Co2Zr2NKADsvGYA8XxuG/4UWgJ6Cjtv+YJnB6MM69QGlQ==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/debug": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.1.tgz", - "integrity": "sha512-KcKCqiftBJcZr++7ykoDIEwSa3XWowTfNPo92BYxjXiyYEVrUQh2aLyhxBCwww+heortUFxEJYcRzosstTEBYQ==", - "dev": true, - "dependencies": { - "ms": "^2.1.3" - }, - "engines": { - "node": ">=6.0" - }, - "peerDependenciesMeta": { - "supports-color": { - "optional": true - } - } - }, - "node_modules/deep-is": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", - "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", - "dev": true - }, - "node_modules/define-data-property": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz", - "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==", - "dev": true, - "peer": true, - "dependencies": { - "es-define-property": "^1.0.0", - "es-errors": "^1.3.0", - "gopd": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/define-properties": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.1.tgz", - "integrity": "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==", - "dev": true, - "peer": true, - "dependencies": { - "define-data-property": "^1.0.1", - "has-property-descriptors": "^1.0.0", - "object-keys": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/doctrine": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", - "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", - "dev": true, - "peer": true, - "dependencies": { - "esutils": "^2.0.2" - }, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/dunder-proto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", - "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind-apply-helpers": "^1.0.1", - "es-errors": "^1.3.0", - "gopd": "^1.2.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-abstract": { - "version": "1.24.0", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.24.0.tgz", - "integrity": "sha512-WSzPgsdLtTcQwm4CROfS5ju2Wa1QQcVeT37jFjYzdFz1r9ahadC8B8/a4qxJxM+09F18iumCdRmlr96ZYkQvEg==", - "dev": true, - "peer": true, - "dependencies": { - "array-buffer-byte-length": "^1.0.2", - "arraybuffer.prototype.slice": "^1.0.4", - "available-typed-arrays": "^1.0.7", - "call-bind": "^1.0.8", - "call-bound": "^1.0.4", - "data-view-buffer": "^1.0.2", - "data-view-byte-length": "^1.0.2", - "data-view-byte-offset": "^1.0.1", - "es-define-property": "^1.0.1", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.1.1", - "es-set-tostringtag": "^2.1.0", - "es-to-primitive": "^1.3.0", - "function.prototype.name": "^1.1.8", - "get-intrinsic": "^1.3.0", - "get-proto": "^1.0.1", - "get-symbol-description": "^1.1.0", - "globalthis": "^1.0.4", - "gopd": "^1.2.0", - "has-property-descriptors": "^1.0.2", - "has-proto": "^1.2.0", - "has-symbols": "^1.1.0", - "hasown": "^2.0.2", - "internal-slot": "^1.1.0", - "is-array-buffer": "^3.0.5", - "is-callable": "^1.2.7", - "is-data-view": "^1.0.2", - "is-negative-zero": "^2.0.3", - "is-regex": "^1.2.1", - "is-set": "^2.0.3", - "is-shared-array-buffer": "^1.0.4", - "is-string": "^1.1.1", - "is-typed-array": "^1.1.15", - "is-weakref": "^1.1.1", - "math-intrinsics": "^1.1.0", - "object-inspect": "^1.13.4", - "object-keys": "^1.1.1", - "object.assign": "^4.1.7", - "own-keys": "^1.0.1", - "regexp.prototype.flags": "^1.5.4", - "safe-array-concat": "^1.1.3", - "safe-push-apply": "^1.0.0", - "safe-regex-test": "^1.1.0", - "set-proto": "^1.0.0", - "stop-iteration-iterator": "^1.1.0", - "string.prototype.trim": "^1.2.10", - "string.prototype.trimend": "^1.0.9", - "string.prototype.trimstart": "^1.0.8", - "typed-array-buffer": "^1.0.3", - "typed-array-byte-length": "^1.0.3", - "typed-array-byte-offset": "^1.0.4", - "typed-array-length": "^1.0.7", - "unbox-primitive": "^1.1.0", - "which-typed-array": "^1.1.19" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/es-define-property": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", - "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-errors": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", - "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-object-atoms": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", - "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", - "dev": true, - "peer": true, - "dependencies": { - "es-errors": "^1.3.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-set-tostringtag": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz", - "integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==", - "dev": true, - "peer": true, - "dependencies": { - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6", - "has-tostringtag": "^1.0.2", - "hasown": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-shim-unscopables": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/es-shim-unscopables/-/es-shim-unscopables-1.1.0.tgz", - "integrity": "sha512-d9T8ucsEhh8Bi1woXCf+TIKDIROLG5WCkxg8geBCbvk22kzwC5G2OnXVMO6FUsvQlgUUXQ2itephWDLqDzbeCw==", - "dev": true, - "peer": true, - "dependencies": { - "hasown": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/es-to-primitive": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.3.0.tgz", - "integrity": "sha512-w+5mJ3GuFL+NjVtJlvydShqE1eN3h3PbI7/5LAsYJP/2qtuMXjfL2LpHSRqo4b4eSF5K/DH1JXKUAHSB2UW50g==", - "dev": true, - "peer": true, - "dependencies": { - "is-callable": "^1.2.7", - "is-date-object": "^1.0.5", - "is-symbol": "^1.0.4" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/escape-string-regexp": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", - "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", - "dev": true, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/eslint": { - "version": "9.28.0", - "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.28.0.tgz", - "integrity": "sha512-ocgh41VhRlf9+fVpe7QKzwLj9c92fDiqOj8Y3Sd4/ZmVA4Btx4PlUYPq4pp9JDyupkf1upbEXecxL2mwNV7jPQ==", - "dev": true, - "dependencies": { - "@eslint-community/eslint-utils": "^4.2.0", - "@eslint-community/regexpp": "^4.12.1", - "@eslint/config-array": "^0.20.0", - "@eslint/config-helpers": "^0.2.1", - "@eslint/core": "^0.14.0", - "@eslint/eslintrc": "^3.3.1", - "@eslint/js": "9.28.0", - "@eslint/plugin-kit": "^0.3.1", - "@humanfs/node": "^0.16.6", - "@humanwhocodes/module-importer": "^1.0.1", - "@humanwhocodes/retry": "^0.4.2", - "@types/estree": "^1.0.6", - "@types/json-schema": "^7.0.15", - "ajv": "^6.12.4", - "chalk": "^4.0.0", - "cross-spawn": "^7.0.6", - "debug": "^4.3.2", - "escape-string-regexp": "^4.0.0", - "eslint-scope": "^8.3.0", - "eslint-visitor-keys": "^4.2.0", - "espree": "^10.3.0", - "esquery": "^1.5.0", - "esutils": "^2.0.2", - "fast-deep-equal": "^3.1.3", - "file-entry-cache": "^8.0.0", - "find-up": "^5.0.0", - "glob-parent": "^6.0.2", - "ignore": "^5.2.0", - "imurmurhash": "^0.1.4", - "is-glob": "^4.0.0", - "json-stable-stringify-without-jsonify": "^1.0.1", - "lodash.merge": "^4.6.2", - "minimatch": "^3.1.2", - "natural-compare": "^1.4.0", - "optionator": "^0.9.3" - }, - "bin": { - "eslint": "bin/eslint.js" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://eslint.org/donate" - }, - "peerDependencies": { - "jiti": "*" - }, - "peerDependenciesMeta": { - "jiti": { - "optional": true - } - } - }, - "node_modules/eslint-config-nodebb": { - "version": "1.1.7", - "resolved": "https://registry.npmjs.org/eslint-config-nodebb/-/eslint-config-nodebb-1.1.7.tgz", - "integrity": "sha512-sdxh4+z0cvzjjpPoubkwl/1+ampG6OERs6vkuM8GzYWkXGBc2vnlBzw4bMpw6Nx6ACe3JBOP3MiVLOBzIGnlMw==", - "dev": true, - "peerDependencies": { - "@eslint/js": "^9.x", - "@stylistic/eslint-plugin": "4.4.1", - "eslint-plugin-import": "2.31.0" - } - }, - "node_modules/eslint-import-resolver-node": { - "version": "0.3.9", - "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.9.tgz", - "integrity": "sha512-WFj2isz22JahUv+B788TlO3N6zL3nNJGU8CcZbPZvVEkBPaJdCV4vy5wyghty5ROFbCRnm132v8BScu5/1BQ8g==", - "dev": true, - "peer": true, - "dependencies": { - "debug": "^3.2.7", - "is-core-module": "^2.13.0", - "resolve": "^1.22.4" - } - }, - "node_modules/eslint-import-resolver-node/node_modules/debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "peer": true, - "dependencies": { - "ms": "^2.1.1" - } - }, - "node_modules/eslint-module-utils": { - "version": "2.12.0", - "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.12.0.tgz", - "integrity": "sha512-wALZ0HFoytlyh/1+4wuZ9FJCD/leWHQzzrxJ8+rebyReSLk7LApMyd3WJaLVoN+D5+WIdJyDK1c6JnE65V4Zyg==", - "dev": true, - "peer": true, - "dependencies": { - "debug": "^3.2.7" - }, - "engines": { - "node": ">=4" - }, - "peerDependenciesMeta": { - "eslint": { - "optional": true - } - } - }, - "node_modules/eslint-module-utils/node_modules/debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "peer": true, - "dependencies": { - "ms": "^2.1.1" - } - }, - "node_modules/eslint-plugin-import": { - "version": "2.31.0", - "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.31.0.tgz", - "integrity": "sha512-ixmkI62Rbc2/w8Vfxyh1jQRTdRTF52VxwRVHl/ykPAmqG+Nb7/kNn+byLP0LxPgI7zWA16Jt82SybJInmMia3A==", - "dev": true, - "peer": true, - "dependencies": { - "@rtsao/scc": "^1.1.0", - "array-includes": "^3.1.8", - "array.prototype.findlastindex": "^1.2.5", - "array.prototype.flat": "^1.3.2", - "array.prototype.flatmap": "^1.3.2", - "debug": "^3.2.7", - "doctrine": "^2.1.0", - "eslint-import-resolver-node": "^0.3.9", - "eslint-module-utils": "^2.12.0", - "hasown": "^2.0.2", - "is-core-module": "^2.15.1", - "is-glob": "^4.0.3", - "minimatch": "^3.1.2", - "object.fromentries": "^2.0.8", - "object.groupby": "^1.0.3", - "object.values": "^1.2.0", - "semver": "^6.3.1", - "string.prototype.trimend": "^1.0.8", - "tsconfig-paths": "^3.15.0" - }, - "engines": { - "node": ">=4" - }, - "peerDependencies": { - "eslint": "^2 || ^3 || ^4 || ^5 || ^6 || ^7.2.0 || ^8 || ^9" - } - }, - "node_modules/eslint-plugin-import/node_modules/debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "peer": true, - "dependencies": { - "ms": "^2.1.1" - } - }, - "node_modules/eslint-plugin-import/node_modules/semver": { - "version": "6.3.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", - "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", - "dev": true, - "peer": true, - "bin": { - "semver": "bin/semver.js" - } - }, - "node_modules/eslint-scope": { - "version": "8.4.0", - "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.4.0.tgz", - "integrity": "sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg==", - "dev": true, - "dependencies": { - "esrecurse": "^4.3.0", - "estraverse": "^5.2.0" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/eslint-visitor-keys": { - "version": "3.4.3", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", - "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", - "dev": true, - "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/eslint/node_modules/eslint-visitor-keys": { - "version": "4.2.1", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.1.tgz", - "integrity": "sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ==", - "dev": true, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/espree": { - "version": "10.4.0", - "resolved": "https://registry.npmjs.org/espree/-/espree-10.4.0.tgz", - "integrity": "sha512-j6PAQ2uUr79PZhBjP5C5fhl8e39FmRnOjsD5lGnWrFU8i2G776tBK7+nP8KuQUTTyAZUwfQqXAgrVH5MbH9CYQ==", - "dev": true, - "dependencies": { - "acorn": "^8.15.0", - "acorn-jsx": "^5.3.2", - "eslint-visitor-keys": "^4.2.1" - }, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/espree/node_modules/eslint-visitor-keys": { - "version": "4.2.1", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.1.tgz", - "integrity": "sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ==", - "dev": true, - "engines": { - "node": "^18.18.0 || ^20.9.0 || >=21.1.0" - }, - "funding": { - "url": "https://opencollective.com/eslint" - } - }, - "node_modules/esquery": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz", - "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==", - "dev": true, - "dependencies": { - "estraverse": "^5.1.0" - }, - "engines": { - "node": ">=0.10" - } - }, - "node_modules/esrecurse": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", - "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", - "dev": true, - "dependencies": { - "estraverse": "^5.2.0" - }, - "engines": { - "node": ">=4.0" - } - }, - "node_modules/estraverse": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", - "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", - "dev": true, - "engines": { - "node": ">=4.0" - } - }, - "node_modules/esutils": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", - "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", - "dev": true, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/fast-deep-equal": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", - "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", - "dev": true - }, - "node_modules/fast-glob": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz", - "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==", - "dev": true, - "peer": true, - "dependencies": { - "@nodelib/fs.stat": "^2.0.2", - "@nodelib/fs.walk": "^1.2.3", - "glob-parent": "^5.1.2", - "merge2": "^1.3.0", - "micromatch": "^4.0.8" - }, - "engines": { - "node": ">=8.6.0" - } - }, - "node_modules/fast-glob/node_modules/glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, - "peer": true, - "dependencies": { - "is-glob": "^4.0.1" - }, - "engines": { - "node": ">= 6" - } - }, - "node_modules/fast-json-stable-stringify": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", - "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", - "dev": true - }, - "node_modules/fast-levenshtein": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", - "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", - "dev": true - }, - "node_modules/fastq": { - "version": "1.19.1", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.1.tgz", - "integrity": "sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==", - "dev": true, - "peer": true, - "dependencies": { - "reusify": "^1.0.4" - } - }, - "node_modules/file-entry-cache": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz", - "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==", - "dev": true, - "dependencies": { - "flat-cache": "^4.0.0" - }, - "engines": { - "node": ">=16.0.0" - } - }, - "node_modules/fill-range": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", - "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", - "dev": true, - "peer": true, - "dependencies": { - "to-regex-range": "^5.0.1" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/find-up": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", - "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", - "dev": true, - "dependencies": { - "locate-path": "^6.0.0", - "path-exists": "^4.0.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/flat-cache": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz", - "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==", - "dev": true, - "dependencies": { - "flatted": "^3.2.9", - "keyv": "^4.5.4" - }, - "engines": { - "node": ">=16" - } - }, - "node_modules/flatted": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.3.tgz", - "integrity": "sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg==", - "dev": true - }, - "node_modules/for-each": { - "version": "0.3.5", - "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.5.tgz", - "integrity": "sha512-dKx12eRCVIzqCxFGplyFKJMPvLEWgmNtUrpTiJIR5u97zEhRG8ySrtboPHZXx7daLxQVrl643cTzbab2tkQjxg==", - "dev": true, - "peer": true, - "dependencies": { - "is-callable": "^1.2.7" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/function-bind": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", - "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", - "dev": true, - "peer": true, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/function.prototype.name": { - "version": "1.1.8", - "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.8.tgz", - "integrity": "sha512-e5iwyodOHhbMr/yNrc7fDYG4qlbIvI5gajyzPnb5TCwyhjApznQh1BMFou9b30SevY43gCJKXycoCBjMbsuW0Q==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "functions-have-names": "^1.2.3", - "hasown": "^2.0.2", - "is-callable": "^1.2.7" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/functions-have-names": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/functions-have-names/-/functions-have-names-1.2.3.tgz", - "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==", - "dev": true, - "peer": true, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/get-intrinsic": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", - "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind-apply-helpers": "^1.0.2", - "es-define-property": "^1.0.1", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.1.1", - "function-bind": "^1.1.2", - "get-proto": "^1.0.1", - "gopd": "^1.2.0", - "has-symbols": "^1.1.0", - "hasown": "^2.0.2", - "math-intrinsics": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/get-proto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", - "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", - "dev": true, - "peer": true, - "dependencies": { - "dunder-proto": "^1.0.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/get-symbol-description": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.1.0.tgz", - "integrity": "sha512-w9UMqWwJxHNOvoNzSJ2oPF5wvYcvP7jUvYzhp67yEhTi17ZDBBC1z9pTdGuzjD+EFIqLSYRweZjqfiPzQ06Ebg==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/glob-parent": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", - "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", - "dev": true, - "dependencies": { - "is-glob": "^4.0.3" - }, - "engines": { - "node": ">=10.13.0" - } - }, - "node_modules/globals": { - "version": "14.0.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-14.0.0.tgz", - "integrity": "sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==", - "dev": true, - "engines": { - "node": ">=18" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/globalthis": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/globalthis/-/globalthis-1.0.4.tgz", - "integrity": "sha512-DpLKbNU4WylpxJykQujfCcwYWiV/Jhm50Goo0wrVILAv5jOr9d+H+UR3PhSCD2rCCEIg0uc+G+muBTwD54JhDQ==", - "dev": true, - "peer": true, - "dependencies": { - "define-properties": "^1.2.1", - "gopd": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/gopd": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", - "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/has-bigints": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.1.0.tgz", - "integrity": "sha512-R3pbpkcIqv2Pm3dUwgjclDRVmWpTJW2DcMzcIhEXEx1oh/CEMObMm3KLmRJOdvhM7o4uQBnwr8pzRK2sJWIqfg==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/has-property-descriptors": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", - "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", - "dev": true, - "peer": true, - "dependencies": { - "es-define-property": "^1.0.0" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/has-proto": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.2.0.tgz", - "integrity": "sha512-KIL7eQPfHQRC8+XluaIw7BHUwwqL19bQn4hzNgdr+1wXoU0KKj6rufu47lhY7KbJR2C6T6+PfyN0Ea7wkSS+qQ==", - "dev": true, - "peer": true, - "dependencies": { - "dunder-proto": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/has-symbols": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", - "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/has-tostringtag": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", - "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", - "dev": true, - "peer": true, - "dependencies": { - "has-symbols": "^1.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/hasown": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", - "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", - "dev": true, - "peer": true, - "dependencies": { - "function-bind": "^1.1.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/ignore": { - "version": "5.3.2", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", - "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", - "dev": true, - "engines": { - "node": ">= 4" - } - }, - "node_modules/import-fresh": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz", - "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==", - "dev": true, - "dependencies": { - "parent-module": "^1.0.0", - "resolve-from": "^4.0.0" - }, - "engines": { - "node": ">=6" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/imurmurhash": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", - "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", - "dev": true, - "engines": { - "node": ">=0.8.19" - } - }, - "node_modules/internal-slot": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.1.0.tgz", - "integrity": "sha512-4gd7VpWNQNB4UKKCFFVcp1AVv+FMOgs9NKzjHKusc8jTMhd5eL1NqQqOpE0KzMds804/yHlglp3uxgluOqAPLw==", - "dev": true, - "peer": true, - "dependencies": { - "es-errors": "^1.3.0", - "hasown": "^2.0.2", - "side-channel": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/is-array-buffer": { - "version": "3.0.5", - "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.5.tgz", - "integrity": "sha512-DDfANUiiG2wC1qawP66qlTugJeL5HyzMpfr8lLK+jMQirGzNod0B12cFB/9q838Ru27sBwfw78/rdoU7RERz6A==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "get-intrinsic": "^1.2.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-async-function": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/is-async-function/-/is-async-function-2.1.1.tgz", - "integrity": "sha512-9dgM/cZBnNvjzaMYHVoxxfPj2QXt22Ev7SuuPrs+xav0ukGB0S6d4ydZdEiM48kLx5kDV+QBPrpVnFyefL8kkQ==", - "dev": true, - "peer": true, - "dependencies": { - "async-function": "^1.0.0", - "call-bound": "^1.0.3", - "get-proto": "^1.0.1", - "has-tostringtag": "^1.0.2", - "safe-regex-test": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-bigint": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.1.0.tgz", - "integrity": "sha512-n4ZT37wG78iz03xPRKJrHTdZbe3IicyucEtdRsV5yglwc3GyUfbAfpSeD0FJ41NbUNSt5wbhqfp1fS+BgnvDFQ==", - "dev": true, - "peer": true, - "dependencies": { - "has-bigints": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-boolean-object": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.2.2.tgz", - "integrity": "sha512-wa56o2/ElJMYqjCjGkXri7it5FbebW5usLw/nPmCMs5DeZ7eziSYZhSmPRn0txqeW4LnAmQQU7FgqLpsEFKM4A==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.3", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-callable": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz", - "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-core-module": { - "version": "2.16.1", - "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", - "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", - "dev": true, - "peer": true, - "dependencies": { - "hasown": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-data-view": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-data-view/-/is-data-view-1.0.2.tgz", - "integrity": "sha512-RKtWF8pGmS87i2D6gqQu/l7EYRlVdfzemCJN/P3UOs//x1QE7mfhvzHIApBTRf7axvT6DMGwSwBXYCT0nfB9xw==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.2", - "get-intrinsic": "^1.2.6", - "is-typed-array": "^1.1.13" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-date-object": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.1.0.tgz", - "integrity": "sha512-PwwhEakHVKTdRNVOw+/Gyh0+MzlCl4R6qKvkhuvLtPMggI1WAHt9sOwZxQLSGpUaDnrdyDsomoRgNnCfKNSXXg==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.2", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-extglob": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", - "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", - "dev": true, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/is-finalizationregistry": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-finalizationregistry/-/is-finalizationregistry-1.1.1.tgz", - "integrity": "sha512-1pC6N8qWJbWoPtEjgcL2xyhQOP491EQjeUo3qTKcmV8YSDDJrOepfG8pcC7h/QgnQHYSv0mJ3Z/ZWxmatVrysg==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-generator-function": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-generator-function/-/is-generator-function-1.1.0.tgz", - "integrity": "sha512-nPUB5km40q9e8UfN/Zc24eLlzdSf9OfKByBw9CIdw4H1giPMeA0OIJvbchsCu4npfI2QcMVBsGEBHKZ7wLTWmQ==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.3", - "get-proto": "^1.0.0", - "has-tostringtag": "^1.0.2", - "safe-regex-test": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-glob": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", - "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", - "dev": true, - "dependencies": { - "is-extglob": "^2.1.1" - }, - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/is-map": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.3.tgz", - "integrity": "sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-negative-zero": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.3.tgz", - "integrity": "sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true, - "peer": true, - "engines": { - "node": ">=0.12.0" - } - }, - "node_modules/is-number-object": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.1.1.tgz", - "integrity": "sha512-lZhclumE1G6VYD8VHe35wFaIif+CTy5SJIi5+3y4psDgWu4wPDoBhF8NxUOinEc7pHgiTsT6MaBb92rKhhD+Xw==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.3", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-regex": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.2.1.tgz", - "integrity": "sha512-MjYsKHO5O7mCsmRGxWcLWheFqN9DJ/2TmngvjKXihe6efViPqc274+Fx/4fYj/r03+ESvBdTXK0V6tA3rgez1g==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.2", - "gopd": "^1.2.0", - "has-tostringtag": "^1.0.2", - "hasown": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-set": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.3.tgz", - "integrity": "sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-shared-array-buffer": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.4.tgz", - "integrity": "sha512-ISWac8drv4ZGfwKl5slpHG9OwPNty4jOWPRIhBpxOoD+hqITiwuipOQ2bNthAzwA3B4fIjO4Nln74N0S9byq8A==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-string": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.1.1.tgz", - "integrity": "sha512-BtEeSsoaQjlSPBemMQIrY1MY0uM6vnS1g5fmufYOtnxLGUZM2178PKbhsk7Ffv58IX+ZtcvoGwccYsh0PglkAA==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.3", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-symbol": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.1.1.tgz", - "integrity": "sha512-9gGx6GTtCQM73BgmHQXfDmLtfjjTUDSyoxTCbp5WtoixAhfgsDirWIcVQ/IHpvI5Vgd5i/J5F7B9cN/WlVbC/w==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.2", - "has-symbols": "^1.1.0", - "safe-regex-test": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-typed-array": { - "version": "1.1.15", - "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.15.tgz", - "integrity": "sha512-p3EcsicXjit7SaskXHs1hA91QxgTw46Fv6EFKKGS5DRFLD8yKnohjF3hxoju94b/OcMZoQukzpPpBE9uLVKzgQ==", - "dev": true, - "peer": true, - "dependencies": { - "which-typed-array": "^1.1.16" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-weakmap": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.2.tgz", - "integrity": "sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-weakref": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.1.1.tgz", - "integrity": "sha512-6i9mGWSlqzNMEqpCp93KwRS1uUOodk2OJ6b+sq7ZPDSy2WuI5NFIxp/254TytR8ftefexkWn5xNiHUNpPOfSew==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/is-weakset": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.4.tgz", - "integrity": "sha512-mfcwb6IzQyOKTs84CQMrOwW4gQcaTOAWJ0zzJCl2WSPDrWk/OzDaImWFH3djXhb24g4eudZfLRozAvPGw4d9hQ==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.3", - "get-intrinsic": "^1.2.6" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/isarray": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", - "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", - "dev": true, - "peer": true - }, - "node_modules/isexe": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", - "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", - "dev": true - }, - "node_modules/js-yaml": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", - "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", - "dev": true, - "dependencies": { - "argparse": "^2.0.1" - }, - "bin": { - "js-yaml": "bin/js-yaml.js" - } - }, - "node_modules/json-buffer": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", - "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", - "dev": true - }, - "node_modules/json-schema-traverse": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", - "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", - "dev": true - }, - "node_modules/json-stable-stringify-without-jsonify": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", - "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", - "dev": true - }, - "node_modules/json5": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.2.tgz", - "integrity": "sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==", - "dev": true, - "peer": true, - "dependencies": { - "minimist": "^1.2.0" - }, - "bin": { - "json5": "lib/cli.js" - } - }, - "node_modules/keyv": { - "version": "4.5.4", - "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", - "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", - "dev": true, - "dependencies": { - "json-buffer": "3.0.1" - } - }, - "node_modules/levn": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", - "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", - "dev": true, - "dependencies": { - "prelude-ls": "^1.2.1", - "type-check": "~0.4.0" - }, - "engines": { - "node": ">= 0.8.0" - } - }, - "node_modules/locate-path": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", - "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", - "dev": true, - "dependencies": { - "p-locate": "^5.0.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/lodash.merge": { - "version": "4.6.2", - "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", - "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", - "dev": true - }, - "node_modules/math-intrinsics": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", - "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/merge2": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", - "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 8" - } - }, - "node_modules/micromatch": { - "version": "4.0.8", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", - "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", - "dev": true, - "peer": true, - "dependencies": { - "braces": "^3.0.3", - "picomatch": "^2.3.1" - }, - "engines": { - "node": ">=8.6" - } - }, - "node_modules/micromatch/node_modules/picomatch": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", - "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", - "dev": true, - "peer": true, - "engines": { - "node": ">=8.6" - }, - "funding": { - "url": "https://github.com/sponsors/jonschlinkert" - } - }, - "node_modules/minimatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", - "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", - "dev": true, - "dependencies": { - "brace-expansion": "^1.1.7" - }, - "engines": { - "node": "*" - } - }, - "node_modules/minimist": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", - "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", - "dev": true, - "peer": true, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/moment": { - "version": "2.29.1", - "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.1.tgz", - "integrity": "sha512-kHmoybcPV8Sqy59DwNDY3Jefr64lK/by/da0ViFcuA4DH0vQg5Q6Ze5VimxkfQNSC+Mls/Kx53s7TjP1RhFEDQ==", - "engines": { - "node": "*" - } - }, - "node_modules/moment-timezone": { - "version": "0.5.32", - "resolved": "https://registry.npmjs.org/moment-timezone/-/moment-timezone-0.5.32.tgz", - "integrity": "sha512-Z8QNyuQHQAmWucp8Knmgei8YNo28aLjJq6Ma+jy1ZSpSk5nyfRT8xgUbSQvD2+2UajISfenndwvFuH3NGS+nvA==", - "dependencies": { - "moment": ">= 2.9.0" - }, - "engines": { - "node": "*" - } - }, - "node_modules/ms": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", - "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", - "dev": true - }, - "node_modules/natural-compare": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", - "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", - "dev": true - }, - "node_modules/object-inspect": { - "version": "1.13.4", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz", - "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/object-keys": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", - "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/object.assign": { - "version": "4.1.7", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.7.tgz", - "integrity": "sha512-nK28WOo+QIjBkDduTINE4JkF/UJJKyf2EJxvJKfblDpyg0Q+pkOHNTL0Qwy6NP6FhE/EnzV73BxxqcJaXY9anw==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0", - "has-symbols": "^1.1.0", - "object-keys": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/object.fromentries": { - "version": "2.0.8", - "resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.8.tgz", - "integrity": "sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.2", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/object.groupby": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/object.groupby/-/object.groupby-1.0.3.tgz", - "integrity": "sha512-+Lhy3TQTuzXI5hevh8sBGqbmurHbbIjAi0Z4S63nthVLmLxfbj4T54a4CfZrXIrt9iP4mVAPYMo/v99taj3wjQ==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/object.values": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.2.1.tgz", - "integrity": "sha512-gXah6aZrcUxjWg2zR2MwouP2eHlCBzdV4pygudehaKXSGW4v2AsRQUK+lwwXhii6KFZcunEnmSUoYp5CXibxtA==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/optionator": { - "version": "0.9.4", - "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", - "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", - "dev": true, - "dependencies": { - "deep-is": "^0.1.3", - "fast-levenshtein": "^2.0.6", - "levn": "^0.4.1", - "prelude-ls": "^1.2.1", - "type-check": "^0.4.0", - "word-wrap": "^1.2.5" - }, - "engines": { - "node": ">= 0.8.0" - } - }, - "node_modules/own-keys": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/own-keys/-/own-keys-1.0.1.tgz", - "integrity": "sha512-qFOyK5PjiWZd+QQIh+1jhdb9LpxTF0qs7Pm8o5QHYZ0M3vKqSqzsZaEB6oWlxZ+q2sJBMI/Ktgd2N5ZwQoRHfg==", - "dev": true, - "peer": true, - "dependencies": { - "get-intrinsic": "^1.2.6", - "object-keys": "^1.1.1", - "safe-push-apply": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/p-limit": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", - "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", - "dev": true, - "dependencies": { - "yocto-queue": "^0.1.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/p-locate": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", - "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", - "dev": true, - "dependencies": { - "p-limit": "^3.0.2" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/parent-module": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", - "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", - "dev": true, - "dependencies": { - "callsites": "^3.0.0" - }, - "engines": { - "node": ">=6" - } - }, - "node_modules/path-exists": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", - "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", - "dev": true, - "engines": { - "node": ">=8" - } - }, - "node_modules/path-key": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", - "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", - "dev": true, - "engines": { - "node": ">=8" - } - }, - "node_modules/path-parse": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", - "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", - "dev": true, - "peer": true - }, - "node_modules/picomatch": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.2.tgz", - "integrity": "sha512-M7BAV6Rlcy5u+m6oPhAPFgJTzAioX/6B0DxyvDlo9l8+T3nLKbrczg2WLUyzd45L8RqfUMyGPzekbMvX2Ldkwg==", - "dev": true, - "peer": true, - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://github.com/sponsors/jonschlinkert" - } - }, - "node_modules/possible-typed-array-names": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.1.0.tgz", - "integrity": "sha512-/+5VFTchJDoVj3bhoqi6UeymcD00DAwb1nJwamzPvHEszJ4FpF6SNNbUbOS8yI56qHzdV8eK0qEfOSiodkTdxg==", - "dev": true, - "peer": true, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/prelude-ls": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", - "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", - "dev": true, - "engines": { - "node": ">= 0.8.0" - } - }, - "node_modules/punycode": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", - "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", - "dev": true, - "engines": { - "node": ">=6" - } - }, - "node_modules/queue-microtask": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", - "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "peer": true - }, - "node_modules/reflect.getprototypeof": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/reflect.getprototypeof/-/reflect.getprototypeof-1.0.10.tgz", - "integrity": "sha512-00o4I+DVrefhv+nX0ulyi3biSHCPDe+yLv5o/p6d/UVlirijB8E16FtfwSAi4g3tcqrQ4lRAqQSoFEZJehYEcw==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.9", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0", - "get-intrinsic": "^1.2.7", - "get-proto": "^1.0.1", - "which-builtin-type": "^1.2.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/regexp.prototype.flags": { - "version": "1.5.4", - "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.4.tgz", - "integrity": "sha512-dYqgNSZbDwkaJ2ceRd9ojCGjBq+mOm9LmtXnAnEGyHhN/5R7iDW2TRw3h+o/jCFxus3P2LfWIIiwowAjANm7IA==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-errors": "^1.3.0", - "get-proto": "^1.0.1", - "gopd": "^1.2.0", - "set-function-name": "^2.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/resolve": { - "version": "1.22.10", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.10.tgz", - "integrity": "sha512-NPRy+/ncIMeDlTAsuqwKIiferiawhefFJtkNSW0qZJEqMEb+qBt/77B/jGeeek+F0uOeN05CDa6HXbbIgtVX4w==", - "dev": true, - "peer": true, - "dependencies": { - "is-core-module": "^2.16.0", - "path-parse": "^1.0.7", - "supports-preserve-symlinks-flag": "^1.0.0" - }, - "bin": { - "resolve": "bin/resolve" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/resolve-from": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", - "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", - "dev": true, - "engines": { - "node": ">=4" - } - }, - "node_modules/reusify": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz", - "integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==", - "dev": true, - "peer": true, - "engines": { - "iojs": ">=1.0.0", - "node": ">=0.10.0" - } - }, - "node_modules/run-parallel": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", - "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", - "dev": true, - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "peer": true, - "dependencies": { - "queue-microtask": "^1.2.2" - } - }, - "node_modules/safe-array-concat": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.1.3.tgz", - "integrity": "sha512-AURm5f0jYEOydBj7VQlVvDrjeFgthDdEF5H1dP+6mNpoXOMo1quQqJ4wvJDyRZ9+pO3kGWoOdmV08cSv2aJV6Q==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.2", - "get-intrinsic": "^1.2.6", - "has-symbols": "^1.1.0", - "isarray": "^2.0.5" - }, - "engines": { - "node": ">=0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/safe-push-apply": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/safe-push-apply/-/safe-push-apply-1.0.0.tgz", - "integrity": "sha512-iKE9w/Z7xCzUMIZqdBsp6pEQvwuEebH4vdpjcDWnyzaI6yl6O9FHvVpmGelvEHNsoY6wGblkxR6Zty/h00WiSA==", - "dev": true, - "peer": true, - "dependencies": { - "es-errors": "^1.3.0", - "isarray": "^2.0.5" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/safe-regex-test": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.1.0.tgz", - "integrity": "sha512-x/+Cz4YrimQxQccJf5mKEbIa1NzeCRNI5Ecl/ekmlYaampdNLPalVyIcCZNNH3MvmqBugV5TMYZXv0ljslUlaw==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "is-regex": "^1.2.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/semver": { - "version": "7.7.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.2.tgz", - "integrity": "sha512-RF0Fw+rO5AMf9MAyaRXI4AV0Ulj5lMHqVxxdSgiVbixSCXoEmmX/jk0CuJw4+3SqroYO9VoUh+HcuJivvtJemA==", - "dev": true, - "peer": true, - "bin": { - "semver": "bin/semver.js" - }, - "engines": { - "node": ">=10" - } - }, - "node_modules/set-function-length": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz", - "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==", - "dev": true, - "peer": true, - "dependencies": { - "define-data-property": "^1.1.4", - "es-errors": "^1.3.0", - "function-bind": "^1.1.2", - "get-intrinsic": "^1.2.4", - "gopd": "^1.0.1", - "has-property-descriptors": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/set-function-name": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.2.tgz", - "integrity": "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==", - "dev": true, - "peer": true, - "dependencies": { - "define-data-property": "^1.1.4", - "es-errors": "^1.3.0", - "functions-have-names": "^1.2.3", - "has-property-descriptors": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/set-proto": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/set-proto/-/set-proto-1.0.0.tgz", - "integrity": "sha512-RJRdvCo6IAnPdsvP/7m6bsQqNnn1FCBX5ZNtFL98MmFF/4xAIJTIg1YbHW5DC2W5SKZanrC6i4HsJqlajw/dZw==", - "dev": true, - "peer": true, - "dependencies": { - "dunder-proto": "^1.0.1", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/shebang-command": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", - "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", - "dev": true, - "dependencies": { - "shebang-regex": "^3.0.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/shebang-regex": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", - "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", - "dev": true, - "engines": { - "node": ">=8" - } - }, - "node_modules/side-channel": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", - "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", - "dev": true, - "peer": true, - "dependencies": { - "es-errors": "^1.3.0", - "object-inspect": "^1.13.3", - "side-channel-list": "^1.0.0", - "side-channel-map": "^1.0.1", - "side-channel-weakmap": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/side-channel-list": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz", - "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==", - "dev": true, - "peer": true, - "dependencies": { - "es-errors": "^1.3.0", - "object-inspect": "^1.13.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/side-channel-map": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", - "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.5", - "object-inspect": "^1.13.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/side-channel-weakmap": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", - "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.5", - "object-inspect": "^1.13.3", - "side-channel-map": "^1.0.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/stop-iteration-iterator": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/stop-iteration-iterator/-/stop-iteration-iterator-1.1.0.tgz", - "integrity": "sha512-eLoXW/DHyl62zxY4SCaIgnRhuMr6ri4juEYARS8E6sCEqzKpOiE521Ucofdx+KnDZl5xmvGYaaKCk5FEOxJCoQ==", - "dev": true, - "peer": true, - "dependencies": { - "es-errors": "^1.3.0", - "internal-slot": "^1.1.0" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/string.prototype.trim": { - "version": "1.2.10", - "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.10.tgz", - "integrity": "sha512-Rs66F0P/1kedk5lyYyH9uBzuiI/kNRmwJAR9quK6VOtIpZ2G+hMZd+HQbbv25MgCA6gEffoMZYxlTod4WcdrKA==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.2", - "define-data-property": "^1.1.4", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-object-atoms": "^1.0.0", - "has-property-descriptors": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/string.prototype.trimend": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.9.tgz", - "integrity": "sha512-G7Ok5C6E/j4SGfyLCloXTrngQIQU3PWtXGst3yM7Bea9FRURf1S42ZHlZZtsNque2FN2PoUhfZXYLNWwEr4dLQ==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.2", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/string.prototype.trimstart": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.8.tgz", - "integrity": "sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/strip-bom": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", - "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==", + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", "dev": true, - "peer": true, "engines": { - "node": ">=4" + "node": ">=8" } }, - "node_modules/strip-json-comments": { + "node_modules/path-key": { "version": "3.1.1", - "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", - "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", "dev": true, "engines": { "node": ">=8" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/supports-preserve-symlinks-flag": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", - "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "node_modules/picomatch": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz", + "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==", "dev": true, "peer": true, "engines": { - "node": ">= 0.4" + "node": ">=12" }, "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/to-regex-range": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", - "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, - "peer": true, - "dependencies": { - "is-number": "^7.0.0" - }, - "engines": { - "node": ">=8.0" - } - }, - "node_modules/ts-api-utils": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.1.0.tgz", - "integrity": "sha512-CUgTZL1irw8u29bzrOD/nH85jqyc74D6SshFgujOIA7osm2Rz7dYH77agkx7H4FBNxDq7Cjf+IjaX/8zwFW+ZQ==", - "dev": true, - "peer": true, - "engines": { - "node": ">=18.12" - }, - "peerDependencies": { - "typescript": ">=4.8.4" - } - }, - "node_modules/tsconfig-paths": { - "version": "3.15.0", - "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.15.0.tgz", - "integrity": "sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg==", - "dev": true, - "peer": true, - "dependencies": { - "@types/json5": "^0.0.29", - "json5": "^1.0.2", - "minimist": "^1.2.6", - "strip-bom": "^3.0.0" + "url": "https://github.com/sponsors/jonschlinkert" } }, - "node_modules/type-check": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", - "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", + "node_modules/prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", "dev": true, - "dependencies": { - "prelude-ls": "^1.2.1" - }, "engines": { "node": ">= 0.8.0" } }, - "node_modules/typed-array-buffer": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.3.tgz", - "integrity": "sha512-nAYYwfY3qnzX30IkA6AQZjVbtK6duGontcQm1WSG1MD94YLqK0515GNApXkoxKOWMusVssAHWLh9SeaoefYFGw==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-typed-array": "^1.1.14" - }, - "engines": { - "node": ">= 0.4" - } - }, - "node_modules/typed-array-byte-length": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.3.tgz", - "integrity": "sha512-BaXgOuIxz8n8pIq3e7Atg/7s+DpiYrxn4vdot3w9KbnBhcRQq6o3xemQdIfynqSeXeDrF32x+WvfzmOjPiY9lg==", - "dev": true, - "peer": true, - "dependencies": { - "call-bind": "^1.0.8", - "for-each": "^0.3.3", - "gopd": "^1.2.0", - "has-proto": "^1.2.0", - "is-typed-array": "^1.1.14" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/typed-array-byte-offset": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.4.tgz", - "integrity": "sha512-bTlAFB/FBYMcuX81gbL4OcpH5PmlFHqlCCpAl8AlEzMz5k53oNDvN8p1PNOWLEmI2x4orp3raOFB51tv9X+MFQ==", + "node_modules/punycode": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", + "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", "dev": true, - "peer": true, - "dependencies": { - "available-typed-arrays": "^1.0.7", - "call-bind": "^1.0.8", - "for-each": "^0.3.3", - "gopd": "^1.2.0", - "has-proto": "^1.2.0", - "is-typed-array": "^1.1.15", - "reflect.getprototypeof": "^1.0.9" - }, "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "node": ">=6" } }, - "node_modules/typed-array-length": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.7.tgz", - "integrity": "sha512-3KS2b+kL7fsuk/eJZ7EQdnEmQoaho/r6KUef7hxvltNA5DR8NAUM+8wJMbJyZ4G9/7i3v5zPBIMN5aybAh2/Jg==", + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", "dev": true, - "peer": true, "dependencies": { - "call-bind": "^1.0.7", - "for-each": "^0.3.3", - "gopd": "^1.0.1", - "is-typed-array": "^1.1.13", - "possible-typed-array-names": "^1.0.0", - "reflect.getprototypeof": "^1.0.6" + "shebang-regex": "^3.0.0" }, "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "node": ">=8" } }, - "node_modules/typescript": { - "version": "5.8.3", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.8.3.tgz", - "integrity": "sha512-p1diW6TqL9L07nNxvRMM7hMMw4c5XOo/1ibL4aAIGmSAt9slTE1Xgw5KWuof2uTOvCg9BY7ZRi+GaF+7sfgPeQ==", + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", "dev": true, - "peer": true, - "bin": { - "tsc": "bin/tsc", - "tsserver": "bin/tsserver" - }, "engines": { - "node": ">=14.17" + "node": ">=8" } }, - "node_modules/unbox-primitive": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.1.0.tgz", - "integrity": "sha512-nWJ91DjeOkej/TA8pXQ3myruKpKEYgqvpw9lz4OPHj/NWFNluYrjbz9j01CJ8yKQd2g4jFoOkINCTW2I5LEEyw==", + "node_modules/type-check": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", + "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", "dev": true, - "peer": true, "dependencies": { - "call-bound": "^1.0.3", - "has-bigints": "^1.0.2", - "has-symbols": "^1.1.0", - "which-boxed-primitive": "^1.1.1" + "prelude-ls": "^1.2.1" }, "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "node": ">= 0.8.0" } }, "node_modules/uri-js": { @@ -3330,95 +960,6 @@ "node": ">= 8" } }, - "node_modules/which-boxed-primitive": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.1.1.tgz", - "integrity": "sha512-TbX3mj8n0odCBFVlY8AxkqcHASw3L60jIuF8jFP78az3C2YhmGvqbHBpAjTRH2/xqYunrJ9g1jSyjCjpoWzIAA==", - "dev": true, - "peer": true, - "dependencies": { - "is-bigint": "^1.1.0", - "is-boolean-object": "^1.2.1", - "is-number-object": "^1.1.1", - "is-string": "^1.1.1", - "is-symbol": "^1.1.1" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/which-builtin-type": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/which-builtin-type/-/which-builtin-type-1.2.1.tgz", - "integrity": "sha512-6iBczoX+kDQ7a3+YJBnh3T+KZRxM/iYNPXicqk66/Qfm1b93iu+yOImkg0zHbj5LNOcNv1TEADiZ0xa34B4q6Q==", - "dev": true, - "peer": true, - "dependencies": { - "call-bound": "^1.0.2", - "function.prototype.name": "^1.1.6", - "has-tostringtag": "^1.0.2", - "is-async-function": "^2.0.0", - "is-date-object": "^1.1.0", - "is-finalizationregistry": "^1.1.0", - "is-generator-function": "^1.0.10", - "is-regex": "^1.2.1", - "is-weakref": "^1.0.2", - "isarray": "^2.0.5", - "which-boxed-primitive": "^1.1.0", - "which-collection": "^1.0.2", - "which-typed-array": "^1.1.16" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/which-collection": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.2.tgz", - "integrity": "sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw==", - "dev": true, - "peer": true, - "dependencies": { - "is-map": "^2.0.3", - "is-set": "^2.0.3", - "is-weakmap": "^2.0.2", - "is-weakset": "^2.0.3" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/which-typed-array": { - "version": "1.1.19", - "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.19.tgz", - "integrity": "sha512-rEvr90Bck4WZt9HHFC4DJMsjvu7x+r6bImz0/BrbWb7A2djJ8hnZMrWnHo9F8ssv0OMErasDhftrfROTyqSDrw==", - "dev": true, - "peer": true, - "dependencies": { - "available-typed-arrays": "^1.0.7", - "call-bind": "^1.0.8", - "call-bound": "^1.0.4", - "for-each": "^0.3.5", - "get-proto": "^1.0.1", - "gopd": "^1.2.0", - "has-tostringtag": "^1.0.2" - }, - "engines": { - "node": ">= 0.4" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/word-wrap": { "version": "1.2.5", "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", @@ -3443,82 +984,70 @@ }, "dependencies": { "@eslint-community/eslint-utils": { - "version": "4.7.0", - "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.7.0.tgz", - "integrity": "sha512-dyybb3AcajC7uha6CvhdVRJqaKyn7w2YKqKyAN37NKYgZT36w+iRb0Dymmc5qEJ549c/S31cMMSFd75bteCpCw==", + "version": "4.9.1", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.1.tgz", + "integrity": "sha512-phrYmNiYppR7znFEdqgfWHXR6NCkZEK7hwWDHZUjit/2/U0r6XvkDl0SYnoM51Hq7FhCGdLDT6zxCCOY1hexsQ==", "dev": true, "requires": { "eslint-visitor-keys": "^3.4.3" } }, "@eslint-community/regexpp": { - "version": "4.12.1", - "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.1.tgz", - "integrity": "sha512-CCZCDJuduB9OUkFkY2IgppNZMi2lBQgD2qzwXkEia16cge2pijY/aXi96CJMquDMn3nJdlPV1A5KrJEXwfLNzQ==", + "version": "4.12.2", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz", + "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==", "dev": true }, "@eslint/config-array": { - "version": "0.20.0", - "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.20.0.tgz", - "integrity": "sha512-fxlS1kkIjx8+vy2SjuCB94q3htSNrufYTXubwiBFeaQHbH6Ipi43gFJq2zCMt6PHhImH3Xmr0NksKDvchWlpQQ==", + "version": "0.23.3", + "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.23.3.tgz", + "integrity": "sha512-j+eEWmB6YYLwcNOdlwQ6L2OsptI/LO6lNBuLIqe5R7RetD658HLoF+Mn7LzYmAWWNNzdC6cqP+L6r8ujeYXWLw==", "dev": true, "requires": { - "@eslint/object-schema": "^2.1.6", + "@eslint/object-schema": "^3.0.3", "debug": "^4.3.1", - "minimatch": "^3.1.2" + "minimatch": "^10.2.4" } }, "@eslint/config-helpers": { - "version": "0.2.2", - "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.2.2.tgz", - "integrity": "sha512-+GPzk8PlG0sPpzdU5ZvIRMPidzAnZDl/s9L+y13iodqvb8leL53bTannOrQ/Im7UkpsmFU5Ily5U60LWixnmLg==", - "dev": true - }, - "@eslint/core": { - "version": "0.14.0", - "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.14.0.tgz", - "integrity": "sha512-qIbV0/JZr7iSDjqAc60IqbLdsj9GDt16xQtWD+B78d/HAlvysGdZZ6rpJHGAc2T0FQx1X6thsSPdnoiGKdNtdg==", + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.5.3.tgz", + "integrity": "sha512-lzGN0onllOZCGroKJmRwY6QcEHxbjBw1gwB8SgRSqK8YbbtEXMvKynsXc3553ckIEBxsbMBU7oOZXKIPGZNeZw==", "dev": true, "requires": { - "@types/json-schema": "^7.0.15" + "@eslint/core": "^1.1.1" } }, - "@eslint/eslintrc": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.3.1.tgz", - "integrity": "sha512-gtF186CXhIl1p4pJNGZw8Yc6RlshoePRvE0X91oPGb3vZ8pM3qOS9W9NGPat9LziaBV7XrJWGylNQXkGcnM3IQ==", + "@eslint/core": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@eslint/core/-/core-1.1.1.tgz", + "integrity": "sha512-QUPblTtE51/7/Zhfv8BDwO0qkkzQL7P/aWWbqcf4xWLEYn1oKjdO0gglQBB4GAsu7u6wjijbCmzsUTy6mnk6oQ==", "dev": true, "requires": { - "ajv": "^6.12.4", - "debug": "^4.3.2", - "espree": "^10.0.1", - "globals": "^14.0.0", - "ignore": "^5.2.0", - "import-fresh": "^3.2.1", - "js-yaml": "^4.1.0", - "minimatch": "^3.1.2", - "strip-json-comments": "^3.1.1" + "@types/json-schema": "^7.0.15" } }, "@eslint/js": { - "version": "9.28.0", - "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.28.0.tgz", - "integrity": "sha512-fnqSjGWd/CoIp4EXIxWVK/sHA6DOHN4+8Ix2cX5ycOY7LG0UY8nHCU5pIp2eaE1Mc7Qd8kHspYNzYXT2ojPLzg==", - "dev": true + "version": "10.0.1", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-10.0.1.tgz", + "integrity": "sha512-zeR9k5pd4gxjZ0abRoIaxdc7I3nDktoXZk2qOv9gCNWx3mVwEn32VRhyLaRsDiJjTs0xq/T8mfPtyuXu7GWBcA==", + "dev": true, + "peer": true, + "requires": {} }, "@eslint/object-schema": { - "version": "2.1.6", - "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.6.tgz", - "integrity": "sha512-RBMg5FRL0I0gs51M/guSAj5/e14VQ4tpZnQNWwuDT66P14I43ItmPfIZRhO9fUVIPOAQXU47atlywZ/czoqFPA==", + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-3.0.3.tgz", + "integrity": "sha512-iM869Pugn9Nsxbh/YHRqYiqd23AmIbxJOcpUMOuWCVNdoQJ5ZtwL6h3t0bcZzJUlC3Dq9jCFCESBZnX0GTv7iQ==", "dev": true }, "@eslint/plugin-kit": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.3.1.tgz", - "integrity": "sha512-0J+zgWxHN+xXONWIyPWKFMgVuJoZuGiIFu8yxk7RJjxkzpGmyja5wRFqZIVtjDVOQpV+Rw0iOAjYPE2eQyjr0w==", + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.6.1.tgz", + "integrity": "sha512-iH1B076HoAshH1mLpHMgwdGeTs0CYwL0SPMkGuSebZrwBp16v415e9NZXg2jtrqPVQjf6IANe2Vtlr5KswtcZQ==", "dev": true, "requires": { - "@eslint/core": "^0.14.0", + "@eslint/core": "^1.1.1", "levn": "^0.4.1" } }, @@ -3534,209 +1063,43 @@ "integrity": "sha512-YuI2ZHQL78Q5HbhDiBA1X4LmYdXCKCMQIfw0pw7piHJwyREFebJUvrQN4cMssyES6x+vfUbx1CIpaQUKYdQZOw==", "dev": true, "requires": { - "@humanfs/core": "^0.19.1", - "@humanwhocodes/retry": "^0.3.0" - }, - "dependencies": { - "@humanwhocodes/retry": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.3.1.tgz", - "integrity": "sha512-JBxkERygn7Bv/GbN5Rv8Ul6LVknS+5Bp6RgDC/O8gEBU/yeH5Ui5C/OlWrTb6qct7LjjfT6Re2NxB0ln0yYybA==", - "dev": true - } - } - }, - "@humanwhocodes/module-importer": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", - "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", - "dev": true - }, - "@humanwhocodes/retry": { - "version": "0.4.3", - "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.3.tgz", - "integrity": "sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==", - "dev": true - }, - "@nodelib/fs.scandir": { - "version": "2.1.5", - "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", - "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", - "dev": true, - "peer": true, - "requires": { - "@nodelib/fs.stat": "2.0.5", - "run-parallel": "^1.1.9" - } - }, - "@nodelib/fs.stat": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", - "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", - "dev": true, - "peer": true - }, - "@nodelib/fs.walk": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", - "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", - "dev": true, - "peer": true, - "requires": { - "@nodelib/fs.scandir": "2.1.5", - "fastq": "^1.6.0" - } - }, - "@rtsao/scc": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@rtsao/scc/-/scc-1.1.0.tgz", - "integrity": "sha512-zt6OdqaDoOnJ1ZYsCYGt9YmWzDXl4vQdKTyJev62gFhRGKdx7mcT54V9KIjg+d2wi9EXsPvAPKe7i7WjfVWB8g==", - "dev": true, - "peer": true - }, - "@stylistic/eslint-plugin": { - "version": "4.4.1", - "resolved": "https://registry.npmjs.org/@stylistic/eslint-plugin/-/eslint-plugin-4.4.1.tgz", - "integrity": "sha512-CEigAk7eOLyHvdgmpZsKFwtiqS2wFwI1fn4j09IU9GmD4euFM4jEBAViWeCqaNLlbX2k2+A/Fq9cje4HQBXuJQ==", - "dev": true, - "peer": true, - "requires": { - "@typescript-eslint/utils": "^8.32.1", - "eslint-visitor-keys": "^4.2.0", - "espree": "^10.3.0", - "estraverse": "^5.3.0", - "picomatch": "^4.0.2" - }, - "dependencies": { - "eslint-visitor-keys": { - "version": "4.2.1", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.1.tgz", - "integrity": "sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ==", - "dev": true, - "peer": true - } - } - }, - "@types/estree": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz", - "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==", - "dev": true - }, - "@types/json-schema": { - "version": "7.0.15", - "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", - "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", - "dev": true - }, - "@types/json5": { - "version": "0.0.29", - "resolved": "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz", - "integrity": "sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==", - "dev": true, - "peer": true - }, - "@typescript-eslint/project-service": { - "version": "8.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/project-service/-/project-service-8.34.0.tgz", - "integrity": "sha512-iEgDALRf970/B2YExmtPMPF54NenZUf4xpL3wsCRx/lgjz6ul/l13R81ozP/ZNuXfnLCS+oPmG7JIxfdNYKELw==", - "dev": true, - "peer": true, - "requires": { - "@typescript-eslint/tsconfig-utils": "^8.34.0", - "@typescript-eslint/types": "^8.34.0", - "debug": "^4.3.4" - } - }, - "@typescript-eslint/scope-manager": { - "version": "8.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.34.0.tgz", - "integrity": "sha512-9Ac0X8WiLykl0aj1oYQNcLZjHgBojT6cW68yAgZ19letYu+Hxd0rE0veI1XznSSst1X5lwnxhPbVdwjDRIomRw==", - "dev": true, - "peer": true, - "requires": { - "@typescript-eslint/types": "8.34.0", - "@typescript-eslint/visitor-keys": "8.34.0" - } - }, - "@typescript-eslint/tsconfig-utils": { - "version": "8.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/tsconfig-utils/-/tsconfig-utils-8.34.0.tgz", - "integrity": "sha512-+W9VYHKFIzA5cBeooqQxqNriAP0QeQ7xTiDuIOr71hzgffm3EL2hxwWBIIj4GuofIbKxGNarpKqIq6Q6YrShOA==", - "dev": true, - "peer": true, - "requires": {} - }, - "@typescript-eslint/types": { - "version": "8.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.34.0.tgz", - "integrity": "sha512-9V24k/paICYPniajHfJ4cuAWETnt7Ssy+R0Rbcqo5sSFr3QEZ/8TSoUi9XeXVBGXCaLtwTOKSLGcInCAvyZeMA==", - "dev": true, - "peer": true - }, - "@typescript-eslint/typescript-estree": { - "version": "8.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.34.0.tgz", - "integrity": "sha512-rOi4KZxI7E0+BMqG7emPSK1bB4RICCpF7QD3KCLXn9ZvWoESsOMlHyZPAHyG04ujVplPaHbmEvs34m+wjgtVtg==", - "dev": true, - "peer": true, - "requires": { - "@typescript-eslint/project-service": "8.34.0", - "@typescript-eslint/tsconfig-utils": "8.34.0", - "@typescript-eslint/types": "8.34.0", - "@typescript-eslint/visitor-keys": "8.34.0", - "debug": "^4.3.4", - "fast-glob": "^3.3.2", - "is-glob": "^4.0.3", - "minimatch": "^9.0.4", - "semver": "^7.6.0", - "ts-api-utils": "^2.1.0" + "@humanfs/core": "^0.19.1", + "@humanwhocodes/retry": "^0.3.0" }, "dependencies": { - "brace-expansion": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", - "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", - "dev": true, - "peer": true, - "requires": { - "balanced-match": "^1.0.0" - } - }, - "minimatch": { - "version": "9.0.5", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", - "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", - "dev": true, - "peer": true, - "requires": { - "brace-expansion": "^2.0.1" - } + "@humanwhocodes/retry": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.3.1.tgz", + "integrity": "sha512-JBxkERygn7Bv/GbN5Rv8Ul6LVknS+5Bp6RgDC/O8gEBU/yeH5Ui5C/OlWrTb6qct7LjjfT6Re2NxB0ln0yYybA==", + "dev": true } } }, - "@typescript-eslint/utils": { - "version": "8.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.34.0.tgz", - "integrity": "sha512-8L4tWatGchV9A1cKbjaavS6mwYwp39jql8xUmIIKJdm+qiaeHy5KMKlBrf30akXAWBzn2SqKsNOtSENWUwg7XQ==", - "dev": true, - "peer": true, - "requires": { - "@eslint-community/eslint-utils": "^4.7.0", - "@typescript-eslint/scope-manager": "8.34.0", - "@typescript-eslint/types": "8.34.0", - "@typescript-eslint/typescript-estree": "8.34.0" - } + "@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true + }, + "@humanwhocodes/retry": { + "version": "0.4.3", + "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.3.tgz", + "integrity": "sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==", + "dev": true }, - "@typescript-eslint/visitor-keys": { - "version": "8.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.34.0.tgz", - "integrity": "sha512-qHV7pW7E85A0x6qyrFn+O+q1k1p3tQCsqIZ1KZ5ESLXY57aTvUd3/a4rdPTeXisvhXn2VQG0VSKUqs8KHF2zcA==", + "@stylistic/eslint-plugin": { + "version": "5.10.0", + "resolved": "https://registry.npmjs.org/@stylistic/eslint-plugin/-/eslint-plugin-5.10.0.tgz", + "integrity": "sha512-nPK52ZHvot8Ju/0A4ucSX1dcPV2/1clx0kLcH5wDmrE4naKso7TUC/voUyU1O9OTKTrR6MYip6LP0ogEMQ9jPQ==", "dev": true, "peer": true, "requires": { - "@typescript-eslint/types": "8.34.0", - "eslint-visitor-keys": "^4.2.0" + "@eslint-community/eslint-utils": "^4.9.1", + "@typescript-eslint/types": "^8.56.0", + "eslint-visitor-keys": "^4.2.1", + "espree": "^10.4.0", + "estraverse": "^5.3.0", + "picomatch": "^4.0.3" }, "dependencies": { "eslint-visitor-keys": { @@ -3748,10 +1111,35 @@ } } }, + "@types/esrecurse": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/@types/esrecurse/-/esrecurse-4.3.1.tgz", + "integrity": "sha512-xJBAbDifo5hpffDBuHl0Y8ywswbiAp/Wi7Y/GtAgSlZyIABppyurxVueOPE8LUQOxdlgi6Zqce7uoEpqNTeiUw==", + "dev": true + }, + "@types/estree": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz", + "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==", + "dev": true + }, + "@types/json-schema": { + "version": "7.0.15", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", + "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", + "dev": true + }, + "@typescript-eslint/types": { + "version": "8.56.1", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.56.1.tgz", + "integrity": "sha512-dbMkdIUkIkchgGDIv7KLUpa0Mda4IYjo4IAMJUZ+3xNoUXxMsk9YtKpTHSChRS85o+H9ftm51gsK1dZReY9CVw==", + "dev": true, + "peer": true + }, "acorn": { - "version": "8.15.0", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.15.0.tgz", - "integrity": "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg==", + "version": "8.16.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.16.0.tgz", + "integrity": "sha512-UVJyE9MttOsBQIDKw1skb9nAwQuR5wuGD3+82K6JgJlm/Y+KI92oNsMNGZCYdDsVtRHSak0pcV5Dno5+4jh9sw==", "dev": true }, "acorn-jsx": { @@ -3762,9 +1150,9 @@ "requires": {} }, "ajv": { - "version": "6.12.6", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", - "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "version": "6.14.0", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.14.0.tgz", + "integrity": "sha512-IWrosm/yrn43eiKqkfkHis7QioDleaXQHdDVPKg0FSwwd/DuvyX79TZnFOnYpB7dcsFAMmtFztZuXPDvSePkFw==", "dev": true, "requires": { "fast-deep-equal": "^3.1.1", @@ -3773,244 +1161,26 @@ "uri-js": "^4.2.2" } }, - "argparse": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", - "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", - "dev": true - }, - "array-buffer-byte-length": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/array-buffer-byte-length/-/array-buffer-byte-length-1.0.2.tgz", - "integrity": "sha512-LHE+8BuR7RYGDKvnrmcuSq3tDcKv9OFEXQt/HpbZhY7V6h0zlUXutnAD82GiFx9rdieCMjkvtcsPqBwgUl1Iiw==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.3", - "is-array-buffer": "^3.0.5" - } - }, - "array-includes": { - "version": "3.1.9", - "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.9.tgz", - "integrity": "sha512-FmeCCAenzH0KH381SPT5FZmiA/TmpndpcaShhfgEN9eCVjnFBqq3l1xrI42y8+PPLI6hypzou4GXw00WHmPBLQ==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.4", - "define-properties": "^1.2.1", - "es-abstract": "^1.24.0", - "es-object-atoms": "^1.1.1", - "get-intrinsic": "^1.3.0", - "is-string": "^1.1.1", - "math-intrinsics": "^1.1.0" - } - }, - "array.prototype.findlastindex": { - "version": "1.2.6", - "resolved": "https://registry.npmjs.org/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.6.tgz", - "integrity": "sha512-F/TKATkzseUExPlfvmwQKGITM3DGTK+vkAsCZoDc5daVygbJBnjEUCbgkAvVFsgfXfX4YIqZ/27G3k3tdXrTxQ==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.4", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.9", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.1.1", - "es-shim-unscopables": "^1.1.0" - } - }, - "array.prototype.flat": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.3.3.tgz", - "integrity": "sha512-rwG/ja1neyLqCuGZ5YYrznA62D4mZXg0i1cIskIUKSiqF3Cje9/wXAls9B9s1Wa2fomMsIv8czB8jZcPmxCXFg==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-shim-unscopables": "^1.0.2" - } - }, - "array.prototype.flatmap": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/array.prototype.flatmap/-/array.prototype.flatmap-1.3.3.tgz", - "integrity": "sha512-Y7Wt51eKJSyi80hFrJCePGGNo5ktJCslFuboqJsbf57CCPcm5zztluPlc4/aD8sWsKvlwatezpV4U1efk8kpjg==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-shim-unscopables": "^1.0.2" - } - }, - "arraybuffer.prototype.slice": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.4.tgz", - "integrity": "sha512-BNoCY6SXXPQ7gF2opIP4GBE+Xw7U+pHMYKuzjgCN3GwiaIR09UUeKfheyIry77QtrCBlC0KK0q5/TER/tYh3PQ==", - "dev": true, - "peer": true, - "requires": { - "array-buffer-byte-length": "^1.0.1", - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6", - "is-array-buffer": "^3.0.4" - } - }, "async": { "version": "3.2.0", "resolved": "https://registry.npmjs.org/async/-/async-3.2.0.tgz", "integrity": "sha512-TR2mEZFVOj2pLStYxLht7TyfuRzaydfpxr3k9RpHIzMgw7A64dzsdqCxH1WJyQdoe8T10nDXd9wnEigmiuHIZw==" }, - "async-function": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/async-function/-/async-function-1.0.0.tgz", - "integrity": "sha512-hsU18Ae8CDTR6Kgu9DYf0EbCr/a5iGL0rytQDobUcdpYOKokk8LEjVphnXkDkgpi0wYVsqrXuP0bZxJaTqdgoA==", - "dev": true, - "peer": true - }, - "available-typed-arrays": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", - "integrity": "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==", - "dev": true, - "peer": true, - "requires": { - "possible-typed-array-names": "^1.0.0" - } - }, "balanced-match": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", - "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-4.0.4.tgz", + "integrity": "sha512-BLrgEcRTwX2o6gGxGOCNyMvGSp35YofuYzw9h1IMTRmKqttAZZVU67bdb9Pr2vUHA8+j3i2tJfjO6C6+4myGTA==", "dev": true }, "brace-expansion": { - "version": "1.1.11", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", - "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", - "dev": true, - "requires": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" - } - }, - "braces": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", - "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", - "dev": true, - "peer": true, - "requires": { - "fill-range": "^7.1.1" - } - }, - "call-bind": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.8.tgz", - "integrity": "sha512-oKlSFMcMwpUg2ednkhQ454wfWiU/ul3CkJe/PEHcTKuiX6RpbehUiFMXu13HalGZxfUwCQzZG747YXBn1im9ww==", - "dev": true, - "peer": true, - "requires": { - "call-bind-apply-helpers": "^1.0.0", - "es-define-property": "^1.0.0", - "get-intrinsic": "^1.2.4", - "set-function-length": "^1.2.2" - } - }, - "call-bind-apply-helpers": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz", - "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==", - "dev": true, - "peer": true, - "requires": { - "es-errors": "^1.3.0", - "function-bind": "^1.1.2" - } - }, - "call-bound": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz", - "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==", - "dev": true, - "peer": true, - "requires": { - "call-bind-apply-helpers": "^1.0.2", - "get-intrinsic": "^1.3.0" - } - }, - "callsites": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", - "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", - "dev": true - }, - "chalk": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.1.tgz", - "integrity": "sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==", + "version": "5.0.4", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-5.0.4.tgz", + "integrity": "sha512-h+DEnpVvxmfVefa4jFbCf5HdH5YMDXRsmKflpf1pILZWRFlTbJpxeU55nJl4Smt5HQaGzg1o6RHFPJaOqnmBDg==", "dev": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", - "dev": true, - "requires": { - "color-convert": "^2.0.1" - } - }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", - "dev": true, - "requires": { - "color-name": "~1.1.4" - } - }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true - }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true - }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", - "dev": true, - "requires": { - "has-flag": "^4.0.0" - } - } + "balanced-match": "^4.0.2" } }, - "concat-map": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", - "dev": true - }, "cron": { "version": "1.8.2", "resolved": "https://registry.npmjs.org/cron/-/cron-1.8.2.tgz", @@ -4030,42 +1200,6 @@ "which": "^2.0.1" } }, - "data-view-buffer": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/data-view-buffer/-/data-view-buffer-1.0.2.tgz", - "integrity": "sha512-EmKO5V3OLXh1rtK2wgXRansaK1/mtVdTUEiEI0W8RkvgT05kfxaH29PliLnpLP73yYO6142Q72QNa8Wx/A5CqQ==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.2" - } - }, - "data-view-byte-length": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/data-view-byte-length/-/data-view-byte-length-1.0.2.tgz", - "integrity": "sha512-tuhGbE6CfTM9+5ANGf+oQb72Ky/0+s3xKUpHvShfiz2RxMFgFPjsXuRLBVMtvMs15awe45SRb83D6wH4ew6wlQ==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.2" - } - }, - "data-view-byte-offset": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/data-view-byte-offset/-/data-view-byte-offset-1.0.1.tgz", - "integrity": "sha512-BS8PfmtDGnrgYdOonGZQdLZslWIeCGFP9tpan0hi1Co2Zr2NKADsvGYA8XxuG/4UWgJ6Cjtv+YJnB6MM69QGlQ==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "is-data-view": "^1.0.1" - } - }, "debug": { "version": "4.4.1", "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.1.tgz", @@ -4081,174 +1215,6 @@ "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", "dev": true }, - "define-data-property": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz", - "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==", - "dev": true, - "peer": true, - "requires": { - "es-define-property": "^1.0.0", - "es-errors": "^1.3.0", - "gopd": "^1.0.1" - } - }, - "define-properties": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.1.tgz", - "integrity": "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==", - "dev": true, - "peer": true, - "requires": { - "define-data-property": "^1.0.1", - "has-property-descriptors": "^1.0.0", - "object-keys": "^1.1.1" - } - }, - "doctrine": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", - "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", - "dev": true, - "peer": true, - "requires": { - "esutils": "^2.0.2" - } - }, - "dunder-proto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", - "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", - "dev": true, - "peer": true, - "requires": { - "call-bind-apply-helpers": "^1.0.1", - "es-errors": "^1.3.0", - "gopd": "^1.2.0" - } - }, - "es-abstract": { - "version": "1.24.0", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.24.0.tgz", - "integrity": "sha512-WSzPgsdLtTcQwm4CROfS5ju2Wa1QQcVeT37jFjYzdFz1r9ahadC8B8/a4qxJxM+09F18iumCdRmlr96ZYkQvEg==", - "dev": true, - "peer": true, - "requires": { - "array-buffer-byte-length": "^1.0.2", - "arraybuffer.prototype.slice": "^1.0.4", - "available-typed-arrays": "^1.0.7", - "call-bind": "^1.0.8", - "call-bound": "^1.0.4", - "data-view-buffer": "^1.0.2", - "data-view-byte-length": "^1.0.2", - "data-view-byte-offset": "^1.0.1", - "es-define-property": "^1.0.1", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.1.1", - "es-set-tostringtag": "^2.1.0", - "es-to-primitive": "^1.3.0", - "function.prototype.name": "^1.1.8", - "get-intrinsic": "^1.3.0", - "get-proto": "^1.0.1", - "get-symbol-description": "^1.1.0", - "globalthis": "^1.0.4", - "gopd": "^1.2.0", - "has-property-descriptors": "^1.0.2", - "has-proto": "^1.2.0", - "has-symbols": "^1.1.0", - "hasown": "^2.0.2", - "internal-slot": "^1.1.0", - "is-array-buffer": "^3.0.5", - "is-callable": "^1.2.7", - "is-data-view": "^1.0.2", - "is-negative-zero": "^2.0.3", - "is-regex": "^1.2.1", - "is-set": "^2.0.3", - "is-shared-array-buffer": "^1.0.4", - "is-string": "^1.1.1", - "is-typed-array": "^1.1.15", - "is-weakref": "^1.1.1", - "math-intrinsics": "^1.1.0", - "object-inspect": "^1.13.4", - "object-keys": "^1.1.1", - "object.assign": "^4.1.7", - "own-keys": "^1.0.1", - "regexp.prototype.flags": "^1.5.4", - "safe-array-concat": "^1.1.3", - "safe-push-apply": "^1.0.0", - "safe-regex-test": "^1.1.0", - "set-proto": "^1.0.0", - "stop-iteration-iterator": "^1.1.0", - "string.prototype.trim": "^1.2.10", - "string.prototype.trimend": "^1.0.9", - "string.prototype.trimstart": "^1.0.8", - "typed-array-buffer": "^1.0.3", - "typed-array-byte-length": "^1.0.3", - "typed-array-byte-offset": "^1.0.4", - "typed-array-length": "^1.0.7", - "unbox-primitive": "^1.1.0", - "which-typed-array": "^1.1.19" - } - }, - "es-define-property": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", - "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", - "dev": true, - "peer": true - }, - "es-errors": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", - "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", - "dev": true, - "peer": true - }, - "es-object-atoms": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", - "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", - "dev": true, - "peer": true, - "requires": { - "es-errors": "^1.3.0" - } - }, - "es-set-tostringtag": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz", - "integrity": "sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA==", - "dev": true, - "peer": true, - "requires": { - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6", - "has-tostringtag": "^1.0.2", - "hasown": "^2.0.2" - } - }, - "es-shim-unscopables": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/es-shim-unscopables/-/es-shim-unscopables-1.1.0.tgz", - "integrity": "sha512-d9T8ucsEhh8Bi1woXCf+TIKDIROLG5WCkxg8geBCbvk22kzwC5G2OnXVMO6FUsvQlgUUXQ2itephWDLqDzbeCw==", - "dev": true, - "peer": true, - "requires": { - "hasown": "^2.0.2" - } - }, - "es-to-primitive": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.3.0.tgz", - "integrity": "sha512-w+5mJ3GuFL+NjVtJlvydShqE1eN3h3PbI7/5LAsYJP/2qtuMXjfL2LpHSRqo4b4eSF5K/DH1JXKUAHSB2UW50g==", - "dev": true, - "peer": true, - "requires": { - "is-callable": "^1.2.7", - "is-date-object": "^1.0.5", - "is-symbol": "^1.0.4" - } - }, "escape-string-regexp": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", @@ -4256,162 +1222,79 @@ "dev": true }, "eslint": { - "version": "9.28.0", - "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.28.0.tgz", - "integrity": "sha512-ocgh41VhRlf9+fVpe7QKzwLj9c92fDiqOj8Y3Sd4/ZmVA4Btx4PlUYPq4pp9JDyupkf1upbEXecxL2mwNV7jPQ==", + "version": "10.0.3", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-10.0.3.tgz", + "integrity": "sha512-COV33RzXZkqhG9P2rZCFl9ZmJ7WL+gQSCRzE7RhkbclbQPtLAWReL7ysA0Sh4c8Im2U9ynybdR56PV0XcKvqaQ==", "dev": true, "requires": { - "@eslint-community/eslint-utils": "^4.2.0", - "@eslint-community/regexpp": "^4.12.1", - "@eslint/config-array": "^0.20.0", - "@eslint/config-helpers": "^0.2.1", - "@eslint/core": "^0.14.0", - "@eslint/eslintrc": "^3.3.1", - "@eslint/js": "9.28.0", - "@eslint/plugin-kit": "^0.3.1", + "@eslint-community/eslint-utils": "^4.8.0", + "@eslint-community/regexpp": "^4.12.2", + "@eslint/config-array": "^0.23.3", + "@eslint/config-helpers": "^0.5.2", + "@eslint/core": "^1.1.1", + "@eslint/plugin-kit": "^0.6.1", "@humanfs/node": "^0.16.6", "@humanwhocodes/module-importer": "^1.0.1", "@humanwhocodes/retry": "^0.4.2", - "@types/estree": "^1.0.6", - "@types/json-schema": "^7.0.15", - "ajv": "^6.12.4", - "chalk": "^4.0.0", - "cross-spawn": "^7.0.6", - "debug": "^4.3.2", - "escape-string-regexp": "^4.0.0", - "eslint-scope": "^8.3.0", - "eslint-visitor-keys": "^4.2.0", - "espree": "^10.3.0", - "esquery": "^1.5.0", - "esutils": "^2.0.2", - "fast-deep-equal": "^3.1.3", - "file-entry-cache": "^8.0.0", - "find-up": "^5.0.0", - "glob-parent": "^6.0.2", - "ignore": "^5.2.0", - "imurmurhash": "^0.1.4", - "is-glob": "^4.0.0", - "json-stable-stringify-without-jsonify": "^1.0.1", - "lodash.merge": "^4.6.2", - "minimatch": "^3.1.2", - "natural-compare": "^1.4.0", - "optionator": "^0.9.3" - }, - "dependencies": { - "eslint-visitor-keys": { - "version": "4.2.1", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.1.tgz", - "integrity": "sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ==", - "dev": true - } - } - }, - "eslint-config-nodebb": { - "version": "1.1.7", - "resolved": "https://registry.npmjs.org/eslint-config-nodebb/-/eslint-config-nodebb-1.1.7.tgz", - "integrity": "sha512-sdxh4+z0cvzjjpPoubkwl/1+ampG6OERs6vkuM8GzYWkXGBc2vnlBzw4bMpw6Nx6ACe3JBOP3MiVLOBzIGnlMw==", - "dev": true, - "requires": {} - }, - "eslint-import-resolver-node": { - "version": "0.3.9", - "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.9.tgz", - "integrity": "sha512-WFj2isz22JahUv+B788TlO3N6zL3nNJGU8CcZbPZvVEkBPaJdCV4vy5wyghty5ROFbCRnm132v8BScu5/1BQ8g==", - "dev": true, - "peer": true, - "requires": { - "debug": "^3.2.7", - "is-core-module": "^2.13.0", - "resolve": "^1.22.4" - }, - "dependencies": { - "debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "peer": true, - "requires": { - "ms": "^2.1.1" - } - } - } - }, - "eslint-module-utils": { - "version": "2.12.0", - "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.12.0.tgz", - "integrity": "sha512-wALZ0HFoytlyh/1+4wuZ9FJCD/leWHQzzrxJ8+rebyReSLk7LApMyd3WJaLVoN+D5+WIdJyDK1c6JnE65V4Zyg==", - "dev": true, - "peer": true, - "requires": { - "debug": "^3.2.7" + "@types/estree": "^1.0.6", + "ajv": "^6.14.0", + "cross-spawn": "^7.0.6", + "debug": "^4.3.2", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^9.1.2", + "eslint-visitor-keys": "^5.0.1", + "espree": "^11.1.1", + "esquery": "^1.7.0", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^8.0.0", + "find-up": "^5.0.0", + "glob-parent": "^6.0.2", + "ignore": "^5.2.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "minimatch": "^10.2.4", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3" }, "dependencies": { - "debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "eslint-visitor-keys": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-5.0.1.tgz", + "integrity": "sha512-tD40eHxA35h0PEIZNeIjkHoDR4YjjJp34biM0mDvplBe//mB+IHCqHDGV7pxF+7MklTvighcCPPZC7ynWyjdTA==", + "dev": true + }, + "espree": { + "version": "11.2.0", + "resolved": "https://registry.npmjs.org/espree/-/espree-11.2.0.tgz", + "integrity": "sha512-7p3DrVEIopW1B1avAGLuCSh1jubc01H2JHc8B4qqGblmg5gI9yumBgACjWo4JlIc04ufug4xJ3SQI8HkS/Rgzw==", "dev": true, - "peer": true, "requires": { - "ms": "^2.1.1" + "acorn": "^8.16.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^5.0.1" } } } }, - "eslint-plugin-import": { - "version": "2.31.0", - "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.31.0.tgz", - "integrity": "sha512-ixmkI62Rbc2/w8Vfxyh1jQRTdRTF52VxwRVHl/ykPAmqG+Nb7/kNn+byLP0LxPgI7zWA16Jt82SybJInmMia3A==", + "eslint-config-nodebb": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/eslint-config-nodebb/-/eslint-config-nodebb-2.0.1.tgz", + "integrity": "sha512-06QoLSEtjcUOstURvS4T2GyTFQOUdD/dCvJkrNi+CQ+9uJPpM41uZQAKJi94gv9OFTeLSdoqxESUXLbw2czqjw==", "dev": true, - "peer": true, "requires": { - "@rtsao/scc": "^1.1.0", - "array-includes": "^3.1.8", - "array.prototype.findlastindex": "^1.2.5", - "array.prototype.flat": "^1.3.2", - "array.prototype.flatmap": "^1.3.2", - "debug": "^3.2.7", - "doctrine": "^2.1.0", - "eslint-import-resolver-node": "^0.3.9", - "eslint-module-utils": "^2.12.0", - "hasown": "^2.0.2", - "is-core-module": "^2.15.1", - "is-glob": "^4.0.3", - "minimatch": "^3.1.2", - "object.fromentries": "^2.0.8", - "object.groupby": "^1.0.3", - "object.values": "^1.2.0", - "semver": "^6.3.1", - "string.prototype.trimend": "^1.0.8", - "tsconfig-paths": "^3.15.0" - }, - "dependencies": { - "debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "peer": true, - "requires": { - "ms": "^2.1.1" - } - }, - "semver": { - "version": "6.3.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", - "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", - "dev": true, - "peer": true - } + "globals": "17.3.0" } }, "eslint-scope": { - "version": "8.4.0", - "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.4.0.tgz", - "integrity": "sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg==", + "version": "9.1.2", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-9.1.2.tgz", + "integrity": "sha512-xS90H51cKw0jltxmvmHy2Iai1LIqrfbw57b79w/J7MfvDfkIkFZ+kj6zC3BjtUwh150HsSSdxXZcsuv72miDFQ==", "dev": true, "requires": { + "@types/esrecurse": "^4.3.1", + "@types/estree": "^1.0.8", "esrecurse": "^4.3.0", "estraverse": "^5.2.0" } @@ -4427,6 +1310,7 @@ "resolved": "https://registry.npmjs.org/espree/-/espree-10.4.0.tgz", "integrity": "sha512-j6PAQ2uUr79PZhBjP5C5fhl8e39FmRnOjsD5lGnWrFU8i2G776tBK7+nP8KuQUTTyAZUwfQqXAgrVH5MbH9CYQ==", "dev": true, + "peer": true, "requires": { "acorn": "^8.15.0", "acorn-jsx": "^5.3.2", @@ -4437,14 +1321,15 @@ "version": "4.2.1", "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.1.tgz", "integrity": "sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ==", - "dev": true + "dev": true, + "peer": true } } }, "esquery": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz", - "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==", + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.7.0.tgz", + "integrity": "sha512-Ap6G0WQwcU/LHsvLwON1fAQX9Zp0A2Y6Y/cJBl9r/JbW90Zyg4/zbG6zzKa2OTALELarYHmKu0GhpM5EO+7T0g==", "dev": true, "requires": { "estraverse": "^5.1.0" @@ -4477,32 +1362,6 @@ "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", "dev": true }, - "fast-glob": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz", - "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==", - "dev": true, - "peer": true, - "requires": { - "@nodelib/fs.stat": "^2.0.2", - "@nodelib/fs.walk": "^1.2.3", - "glob-parent": "^5.1.2", - "merge2": "^1.3.0", - "micromatch": "^4.0.8" - }, - "dependencies": { - "glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, - "peer": true, - "requires": { - "is-glob": "^4.0.1" - } - } - } - }, "fast-json-stable-stringify": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", @@ -4515,16 +1374,6 @@ "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", "dev": true }, - "fastq": { - "version": "1.19.1", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.1.tgz", - "integrity": "sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==", - "dev": true, - "peer": true, - "requires": { - "reusify": "^1.0.4" - } - }, "file-entry-cache": { "version": "8.0.0", "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz", @@ -4534,16 +1383,6 @@ "flat-cache": "^4.0.0" } }, - "fill-range": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", - "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", - "dev": true, - "peer": true, - "requires": { - "to-regex-range": "^5.0.1" - } - }, "find-up": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", @@ -4570,87 +1409,6 @@ "integrity": "sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg==", "dev": true }, - "for-each": { - "version": "0.3.5", - "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.5.tgz", - "integrity": "sha512-dKx12eRCVIzqCxFGplyFKJMPvLEWgmNtUrpTiJIR5u97zEhRG8ySrtboPHZXx7daLxQVrl643cTzbab2tkQjxg==", - "dev": true, - "peer": true, - "requires": { - "is-callable": "^1.2.7" - } - }, - "function-bind": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", - "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", - "dev": true, - "peer": true - }, - "function.prototype.name": { - "version": "1.1.8", - "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.8.tgz", - "integrity": "sha512-e5iwyodOHhbMr/yNrc7fDYG4qlbIvI5gajyzPnb5TCwyhjApznQh1BMFou9b30SevY43gCJKXycoCBjMbsuW0Q==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "functions-have-names": "^1.2.3", - "hasown": "^2.0.2", - "is-callable": "^1.2.7" - } - }, - "functions-have-names": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/functions-have-names/-/functions-have-names-1.2.3.tgz", - "integrity": "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ==", - "dev": true, - "peer": true - }, - "get-intrinsic": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", - "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", - "dev": true, - "peer": true, - "requires": { - "call-bind-apply-helpers": "^1.0.2", - "es-define-property": "^1.0.1", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.1.1", - "function-bind": "^1.1.2", - "get-proto": "^1.0.1", - "gopd": "^1.2.0", - "has-symbols": "^1.1.0", - "hasown": "^2.0.2", - "math-intrinsics": "^1.1.0" - } - }, - "get-proto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", - "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", - "dev": true, - "peer": true, - "requires": { - "dunder-proto": "^1.0.1", - "es-object-atoms": "^1.0.0" - } - }, - "get-symbol-description": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.1.0.tgz", - "integrity": "sha512-w9UMqWwJxHNOvoNzSJ2oPF5wvYcvP7jUvYzhp67yEhTi17ZDBBC1z9pTdGuzjD+EFIqLSYRweZjqfiPzQ06Ebg==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.6" - } - }, "glob-parent": { "version": "6.0.2", "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", @@ -4661,233 +1419,29 @@ } }, "globals": { - "version": "14.0.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-14.0.0.tgz", - "integrity": "sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==", + "version": "17.3.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-17.3.0.tgz", + "integrity": "sha512-yMqGUQVVCkD4tqjOJf3TnrvaaHDMYp4VlUSObbkIiuCPe/ofdMBFIAcBbCSRFWOnos6qRiTVStDwqPLUclaxIw==", "dev": true }, - "globalthis": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/globalthis/-/globalthis-1.0.4.tgz", - "integrity": "sha512-DpLKbNU4WylpxJykQujfCcwYWiV/Jhm50Goo0wrVILAv5jOr9d+H+UR3PhSCD2rCCEIg0uc+G+muBTwD54JhDQ==", - "dev": true, - "peer": true, - "requires": { - "define-properties": "^1.2.1", - "gopd": "^1.0.1" - } - }, - "gopd": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", - "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", - "dev": true, - "peer": true - }, - "has-bigints": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.1.0.tgz", - "integrity": "sha512-R3pbpkcIqv2Pm3dUwgjclDRVmWpTJW2DcMzcIhEXEx1oh/CEMObMm3KLmRJOdvhM7o4uQBnwr8pzRK2sJWIqfg==", - "dev": true, - "peer": true - }, - "has-property-descriptors": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", - "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", - "dev": true, - "peer": true, - "requires": { - "es-define-property": "^1.0.0" - } - }, - "has-proto": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.2.0.tgz", - "integrity": "sha512-KIL7eQPfHQRC8+XluaIw7BHUwwqL19bQn4hzNgdr+1wXoU0KKj6rufu47lhY7KbJR2C6T6+PfyN0Ea7wkSS+qQ==", - "dev": true, - "peer": true, - "requires": { - "dunder-proto": "^1.0.0" - } - }, - "has-symbols": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", - "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", - "dev": true, - "peer": true - }, - "has-tostringtag": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", - "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", - "dev": true, - "peer": true, - "requires": { - "has-symbols": "^1.0.3" - } - }, - "hasown": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", - "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", - "dev": true, - "peer": true, - "requires": { - "function-bind": "^1.1.2" - } - }, "ignore": { "version": "5.3.2", "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", "dev": true }, - "import-fresh": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz", - "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==", - "dev": true, - "requires": { - "parent-module": "^1.0.0", - "resolve-from": "^4.0.0" - } - }, "imurmurhash": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", "dev": true }, - "internal-slot": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.1.0.tgz", - "integrity": "sha512-4gd7VpWNQNB4UKKCFFVcp1AVv+FMOgs9NKzjHKusc8jTMhd5eL1NqQqOpE0KzMds804/yHlglp3uxgluOqAPLw==", - "dev": true, - "peer": true, - "requires": { - "es-errors": "^1.3.0", - "hasown": "^2.0.2", - "side-channel": "^1.1.0" - } - }, - "is-array-buffer": { - "version": "3.0.5", - "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.5.tgz", - "integrity": "sha512-DDfANUiiG2wC1qawP66qlTugJeL5HyzMpfr8lLK+jMQirGzNod0B12cFB/9q838Ru27sBwfw78/rdoU7RERz6A==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "get-intrinsic": "^1.2.6" - } - }, - "is-async-function": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/is-async-function/-/is-async-function-2.1.1.tgz", - "integrity": "sha512-9dgM/cZBnNvjzaMYHVoxxfPj2QXt22Ev7SuuPrs+xav0ukGB0S6d4ydZdEiM48kLx5kDV+QBPrpVnFyefL8kkQ==", - "dev": true, - "peer": true, - "requires": { - "async-function": "^1.0.0", - "call-bound": "^1.0.3", - "get-proto": "^1.0.1", - "has-tostringtag": "^1.0.2", - "safe-regex-test": "^1.1.0" - } - }, - "is-bigint": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.1.0.tgz", - "integrity": "sha512-n4ZT37wG78iz03xPRKJrHTdZbe3IicyucEtdRsV5yglwc3GyUfbAfpSeD0FJ41NbUNSt5wbhqfp1fS+BgnvDFQ==", - "dev": true, - "peer": true, - "requires": { - "has-bigints": "^1.0.2" - } - }, - "is-boolean-object": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.2.2.tgz", - "integrity": "sha512-wa56o2/ElJMYqjCjGkXri7it5FbebW5usLw/nPmCMs5DeZ7eziSYZhSmPRn0txqeW4LnAmQQU7FgqLpsEFKM4A==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.3", - "has-tostringtag": "^1.0.2" - } - }, - "is-callable": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.7.tgz", - "integrity": "sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA==", - "dev": true, - "peer": true - }, - "is-core-module": { - "version": "2.16.1", - "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", - "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", - "dev": true, - "peer": true, - "requires": { - "hasown": "^2.0.2" - } - }, - "is-data-view": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-data-view/-/is-data-view-1.0.2.tgz", - "integrity": "sha512-RKtWF8pGmS87i2D6gqQu/l7EYRlVdfzemCJN/P3UOs//x1QE7mfhvzHIApBTRf7axvT6DMGwSwBXYCT0nfB9xw==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.2", - "get-intrinsic": "^1.2.6", - "is-typed-array": "^1.1.13" - } - }, - "is-date-object": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.1.0.tgz", - "integrity": "sha512-PwwhEakHVKTdRNVOw+/Gyh0+MzlCl4R6qKvkhuvLtPMggI1WAHt9sOwZxQLSGpUaDnrdyDsomoRgNnCfKNSXXg==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.2", - "has-tostringtag": "^1.0.2" - } - }, "is-extglob": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", "dev": true }, - "is-finalizationregistry": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-finalizationregistry/-/is-finalizationregistry-1.1.1.tgz", - "integrity": "sha512-1pC6N8qWJbWoPtEjgcL2xyhQOP491EQjeUo3qTKcmV8YSDDJrOepfG8pcC7h/QgnQHYSv0mJ3Z/ZWxmatVrysg==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.3" - } - }, - "is-generator-function": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-generator-function/-/is-generator-function-1.1.0.tgz", - "integrity": "sha512-nPUB5km40q9e8UfN/Zc24eLlzdSf9OfKByBw9CIdw4H1giPMeA0OIJvbchsCu4npfI2QcMVBsGEBHKZ7wLTWmQ==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.3", - "get-proto": "^1.0.0", - "has-tostringtag": "^1.0.2", - "safe-regex-test": "^1.1.0" - } - }, "is-glob": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", @@ -4897,150 +1451,11 @@ "is-extglob": "^2.1.1" } }, - "is-map": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.3.tgz", - "integrity": "sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw==", - "dev": true, - "peer": true - }, - "is-negative-zero": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.3.tgz", - "integrity": "sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==", - "dev": true, - "peer": true - }, - "is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true, - "peer": true - }, - "is-number-object": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.1.1.tgz", - "integrity": "sha512-lZhclumE1G6VYD8VHe35wFaIif+CTy5SJIi5+3y4psDgWu4wPDoBhF8NxUOinEc7pHgiTsT6MaBb92rKhhD+Xw==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.3", - "has-tostringtag": "^1.0.2" - } - }, - "is-regex": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.2.1.tgz", - "integrity": "sha512-MjYsKHO5O7mCsmRGxWcLWheFqN9DJ/2TmngvjKXihe6efViPqc274+Fx/4fYj/r03+ESvBdTXK0V6tA3rgez1g==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.2", - "gopd": "^1.2.0", - "has-tostringtag": "^1.0.2", - "hasown": "^2.0.2" - } - }, - "is-set": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.3.tgz", - "integrity": "sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg==", - "dev": true, - "peer": true - }, - "is-shared-array-buffer": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.4.tgz", - "integrity": "sha512-ISWac8drv4ZGfwKl5slpHG9OwPNty4jOWPRIhBpxOoD+hqITiwuipOQ2bNthAzwA3B4fIjO4Nln74N0S9byq8A==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.3" - } - }, - "is-string": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.1.1.tgz", - "integrity": "sha512-BtEeSsoaQjlSPBemMQIrY1MY0uM6vnS1g5fmufYOtnxLGUZM2178PKbhsk7Ffv58IX+ZtcvoGwccYsh0PglkAA==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.3", - "has-tostringtag": "^1.0.2" - } - }, - "is-symbol": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.1.1.tgz", - "integrity": "sha512-9gGx6GTtCQM73BgmHQXfDmLtfjjTUDSyoxTCbp5WtoixAhfgsDirWIcVQ/IHpvI5Vgd5i/J5F7B9cN/WlVbC/w==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.2", - "has-symbols": "^1.1.0", - "safe-regex-test": "^1.1.0" - } - }, - "is-typed-array": { - "version": "1.1.15", - "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.15.tgz", - "integrity": "sha512-p3EcsicXjit7SaskXHs1hA91QxgTw46Fv6EFKKGS5DRFLD8yKnohjF3hxoju94b/OcMZoQukzpPpBE9uLVKzgQ==", - "dev": true, - "peer": true, - "requires": { - "which-typed-array": "^1.1.16" - } - }, - "is-weakmap": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.2.tgz", - "integrity": "sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w==", - "dev": true, - "peer": true - }, - "is-weakref": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.1.1.tgz", - "integrity": "sha512-6i9mGWSlqzNMEqpCp93KwRS1uUOodk2OJ6b+sq7ZPDSy2WuI5NFIxp/254TytR8ftefexkWn5xNiHUNpPOfSew==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.3" - } - }, - "is-weakset": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.4.tgz", - "integrity": "sha512-mfcwb6IzQyOKTs84CQMrOwW4gQcaTOAWJ0zzJCl2WSPDrWk/OzDaImWFH3djXhb24g4eudZfLRozAvPGw4d9hQ==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.3", - "get-intrinsic": "^1.2.6" - } - }, - "isarray": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", - "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", - "dev": true, - "peer": true - }, "isexe": { "version": "2.0.0", - "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", - "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", - "dev": true - }, - "js-yaml": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", - "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", - "dev": true, - "requires": { - "argparse": "^2.0.1" - } + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true }, "json-buffer": { "version": "3.0.1", @@ -5060,16 +1475,6 @@ "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", "dev": true }, - "json5": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.2.tgz", - "integrity": "sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==", - "dev": true, - "peer": true, - "requires": { - "minimist": "^1.2.0" - } - }, "keyv": { "version": "4.5.4", "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", @@ -5098,62 +1503,15 @@ "p-locate": "^5.0.0" } }, - "lodash.merge": { - "version": "4.6.2", - "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", - "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", - "dev": true - }, - "math-intrinsics": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", - "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", - "dev": true, - "peer": true - }, - "merge2": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", - "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", - "dev": true, - "peer": true - }, - "micromatch": { - "version": "4.0.8", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", - "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", - "dev": true, - "peer": true, - "requires": { - "braces": "^3.0.3", - "picomatch": "^2.3.1" - }, - "dependencies": { - "picomatch": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", - "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", - "dev": true, - "peer": true - } - } - }, "minimatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", - "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "version": "10.2.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.4.tgz", + "integrity": "sha512-oRjTw/97aTBN0RHbYCdtF1MQfvusSIBQM0IZEgzl6426+8jSC0nF1a/GmnVLpfB9yyr6g6FTqWqiZVbxrtaCIg==", "dev": true, "requires": { - "brace-expansion": "^1.1.7" + "brace-expansion": "^5.0.2" } }, - "minimist": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", - "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", - "dev": true, - "peer": true - }, "moment": { "version": "2.29.1", "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.1.tgz", @@ -5179,73 +1537,6 @@ "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", "dev": true }, - "object-inspect": { - "version": "1.13.4", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz", - "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==", - "dev": true, - "peer": true - }, - "object-keys": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", - "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", - "dev": true, - "peer": true - }, - "object.assign": { - "version": "4.1.7", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.7.tgz", - "integrity": "sha512-nK28WOo+QIjBkDduTINE4JkF/UJJKyf2EJxvJKfblDpyg0Q+pkOHNTL0Qwy6NP6FhE/EnzV73BxxqcJaXY9anw==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0", - "has-symbols": "^1.1.0", - "object-keys": "^1.1.1" - } - }, - "object.fromentries": { - "version": "2.0.8", - "resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.8.tgz", - "integrity": "sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.2", - "es-object-atoms": "^1.0.0" - } - }, - "object.groupby": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/object.groupby/-/object.groupby-1.0.3.tgz", - "integrity": "sha512-+Lhy3TQTuzXI5hevh8sBGqbmurHbbIjAi0Z4S63nthVLmLxfbj4T54a4CfZrXIrt9iP4mVAPYMo/v99taj3wjQ==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.2" - } - }, - "object.values": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.2.1.tgz", - "integrity": "sha512-gXah6aZrcUxjWg2zR2MwouP2eHlCBzdV4pygudehaKXSGW4v2AsRQUK+lwwXhii6KFZcunEnmSUoYp5CXibxtA==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.3", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - } - }, "optionator": { "version": "0.9.4", "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", @@ -5260,18 +1551,6 @@ "word-wrap": "^1.2.5" } }, - "own-keys": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/own-keys/-/own-keys-1.0.1.tgz", - "integrity": "sha512-qFOyK5PjiWZd+QQIh+1jhdb9LpxTF0qs7Pm8o5QHYZ0M3vKqSqzsZaEB6oWlxZ+q2sJBMI/Ktgd2N5ZwQoRHfg==", - "dev": true, - "peer": true, - "requires": { - "get-intrinsic": "^1.2.6", - "object-keys": "^1.1.1", - "safe-push-apply": "^1.0.0" - } - }, "p-limit": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", @@ -5290,15 +1569,6 @@ "p-limit": "^3.0.2" } }, - "parent-module": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", - "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", - "dev": true, - "requires": { - "callsites": "^3.0.0" - } - }, "path-exists": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", @@ -5311,24 +1581,10 @@ "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", "dev": true }, - "path-parse": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", - "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", - "dev": true, - "peer": true - }, "picomatch": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.2.tgz", - "integrity": "sha512-M7BAV6Rlcy5u+m6oPhAPFgJTzAioX/6B0DxyvDlo9l8+T3nLKbrczg2WLUyzd45L8RqfUMyGPzekbMvX2Ldkwg==", - "dev": true, - "peer": true - }, - "possible-typed-array-names": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.1.0.tgz", - "integrity": "sha512-/+5VFTchJDoVj3bhoqi6UeymcD00DAwb1nJwamzPvHEszJ4FpF6SNNbUbOS8yI56qHzdV8eK0qEfOSiodkTdxg==", + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.3.tgz", + "integrity": "sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==", "dev": true, "peer": true }, @@ -5344,164 +1600,6 @@ "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", "dev": true }, - "queue-microtask": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", - "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", - "dev": true, - "peer": true - }, - "reflect.getprototypeof": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/reflect.getprototypeof/-/reflect.getprototypeof-1.0.10.tgz", - "integrity": "sha512-00o4I+DVrefhv+nX0ulyi3biSHCPDe+yLv5o/p6d/UVlirijB8E16FtfwSAi4g3tcqrQ4lRAqQSoFEZJehYEcw==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.9", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0", - "get-intrinsic": "^1.2.7", - "get-proto": "^1.0.1", - "which-builtin-type": "^1.2.1" - } - }, - "regexp.prototype.flags": { - "version": "1.5.4", - "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.4.tgz", - "integrity": "sha512-dYqgNSZbDwkaJ2ceRd9ojCGjBq+mOm9LmtXnAnEGyHhN/5R7iDW2TRw3h+o/jCFxus3P2LfWIIiwowAjANm7IA==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.8", - "define-properties": "^1.2.1", - "es-errors": "^1.3.0", - "get-proto": "^1.0.1", - "gopd": "^1.2.0", - "set-function-name": "^2.0.2" - } - }, - "resolve": { - "version": "1.22.10", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.10.tgz", - "integrity": "sha512-NPRy+/ncIMeDlTAsuqwKIiferiawhefFJtkNSW0qZJEqMEb+qBt/77B/jGeeek+F0uOeN05CDa6HXbbIgtVX4w==", - "dev": true, - "peer": true, - "requires": { - "is-core-module": "^2.16.0", - "path-parse": "^1.0.7", - "supports-preserve-symlinks-flag": "^1.0.0" - } - }, - "resolve-from": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", - "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", - "dev": true - }, - "reusify": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz", - "integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==", - "dev": true, - "peer": true - }, - "run-parallel": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", - "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", - "dev": true, - "peer": true, - "requires": { - "queue-microtask": "^1.2.2" - } - }, - "safe-array-concat": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.1.3.tgz", - "integrity": "sha512-AURm5f0jYEOydBj7VQlVvDrjeFgthDdEF5H1dP+6mNpoXOMo1quQqJ4wvJDyRZ9+pO3kGWoOdmV08cSv2aJV6Q==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.2", - "get-intrinsic": "^1.2.6", - "has-symbols": "^1.1.0", - "isarray": "^2.0.5" - } - }, - "safe-push-apply": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/safe-push-apply/-/safe-push-apply-1.0.0.tgz", - "integrity": "sha512-iKE9w/Z7xCzUMIZqdBsp6pEQvwuEebH4vdpjcDWnyzaI6yl6O9FHvVpmGelvEHNsoY6wGblkxR6Zty/h00WiSA==", - "dev": true, - "peer": true, - "requires": { - "es-errors": "^1.3.0", - "isarray": "^2.0.5" - } - }, - "safe-regex-test": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.1.0.tgz", - "integrity": "sha512-x/+Cz4YrimQxQccJf5mKEbIa1NzeCRNI5Ecl/ekmlYaampdNLPalVyIcCZNNH3MvmqBugV5TMYZXv0ljslUlaw==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "is-regex": "^1.2.1" - } - }, - "semver": { - "version": "7.7.2", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.2.tgz", - "integrity": "sha512-RF0Fw+rO5AMf9MAyaRXI4AV0Ulj5lMHqVxxdSgiVbixSCXoEmmX/jk0CuJw4+3SqroYO9VoUh+HcuJivvtJemA==", - "dev": true, - "peer": true - }, - "set-function-length": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz", - "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==", - "dev": true, - "peer": true, - "requires": { - "define-data-property": "^1.1.4", - "es-errors": "^1.3.0", - "function-bind": "^1.1.2", - "get-intrinsic": "^1.2.4", - "gopd": "^1.0.1", - "has-property-descriptors": "^1.0.2" - } - }, - "set-function-name": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.2.tgz", - "integrity": "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==", - "dev": true, - "peer": true, - "requires": { - "define-data-property": "^1.1.4", - "es-errors": "^1.3.0", - "functions-have-names": "^1.2.3", - "has-property-descriptors": "^1.0.2" - } - }, - "set-proto": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/set-proto/-/set-proto-1.0.0.tgz", - "integrity": "sha512-RJRdvCo6IAnPdsvP/7m6bsQqNnn1FCBX5ZNtFL98MmFF/4xAIJTIg1YbHW5DC2W5SKZanrC6i4HsJqlajw/dZw==", - "dev": true, - "peer": true, - "requires": { - "dunder-proto": "^1.0.1", - "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0" - } - }, "shebang-command": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", @@ -5517,161 +1615,6 @@ "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", "dev": true }, - "side-channel": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", - "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", - "dev": true, - "peer": true, - "requires": { - "es-errors": "^1.3.0", - "object-inspect": "^1.13.3", - "side-channel-list": "^1.0.0", - "side-channel-map": "^1.0.1", - "side-channel-weakmap": "^1.0.2" - } - }, - "side-channel-list": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz", - "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==", - "dev": true, - "peer": true, - "requires": { - "es-errors": "^1.3.0", - "object-inspect": "^1.13.3" - } - }, - "side-channel-map": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", - "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.5", - "object-inspect": "^1.13.3" - } - }, - "side-channel-weakmap": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", - "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.2", - "es-errors": "^1.3.0", - "get-intrinsic": "^1.2.5", - "object-inspect": "^1.13.3", - "side-channel-map": "^1.0.1" - } - }, - "stop-iteration-iterator": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/stop-iteration-iterator/-/stop-iteration-iterator-1.1.0.tgz", - "integrity": "sha512-eLoXW/DHyl62zxY4SCaIgnRhuMr6ri4juEYARS8E6sCEqzKpOiE521Ucofdx+KnDZl5xmvGYaaKCk5FEOxJCoQ==", - "dev": true, - "peer": true, - "requires": { - "es-errors": "^1.3.0", - "internal-slot": "^1.1.0" - } - }, - "string.prototype.trim": { - "version": "1.2.10", - "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.10.tgz", - "integrity": "sha512-Rs66F0P/1kedk5lyYyH9uBzuiI/kNRmwJAR9quK6VOtIpZ2G+hMZd+HQbbv25MgCA6gEffoMZYxlTod4WcdrKA==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.2", - "define-data-property": "^1.1.4", - "define-properties": "^1.2.1", - "es-abstract": "^1.23.5", - "es-object-atoms": "^1.0.0", - "has-property-descriptors": "^1.0.2" - } - }, - "string.prototype.trimend": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.9.tgz", - "integrity": "sha512-G7Ok5C6E/j4SGfyLCloXTrngQIQU3PWtXGst3yM7Bea9FRURf1S42ZHlZZtsNque2FN2PoUhfZXYLNWwEr4dLQ==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.8", - "call-bound": "^1.0.2", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - } - }, - "string.prototype.trimstart": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.8.tgz", - "integrity": "sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.7", - "define-properties": "^1.2.1", - "es-object-atoms": "^1.0.0" - } - }, - "strip-bom": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", - "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==", - "dev": true, - "peer": true - }, - "strip-json-comments": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", - "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", - "dev": true - }, - "supports-preserve-symlinks-flag": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", - "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", - "dev": true, - "peer": true - }, - "to-regex-range": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", - "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, - "peer": true, - "requires": { - "is-number": "^7.0.0" - } - }, - "ts-api-utils": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.1.0.tgz", - "integrity": "sha512-CUgTZL1irw8u29bzrOD/nH85jqyc74D6SshFgujOIA7osm2Rz7dYH77agkx7H4FBNxDq7Cjf+IjaX/8zwFW+ZQ==", - "dev": true, - "peer": true, - "requires": {} - }, - "tsconfig-paths": { - "version": "3.15.0", - "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.15.0.tgz", - "integrity": "sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg==", - "dev": true, - "peer": true, - "requires": { - "@types/json5": "^0.0.29", - "json5": "^1.0.2", - "minimist": "^1.2.6", - "strip-bom": "^3.0.0" - } - }, "type-check": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", @@ -5681,83 +1624,6 @@ "prelude-ls": "^1.2.1" } }, - "typed-array-buffer": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.3.tgz", - "integrity": "sha512-nAYYwfY3qnzX30IkA6AQZjVbtK6duGontcQm1WSG1MD94YLqK0515GNApXkoxKOWMusVssAHWLh9SeaoefYFGw==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.3", - "es-errors": "^1.3.0", - "is-typed-array": "^1.1.14" - } - }, - "typed-array-byte-length": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.3.tgz", - "integrity": "sha512-BaXgOuIxz8n8pIq3e7Atg/7s+DpiYrxn4vdot3w9KbnBhcRQq6o3xemQdIfynqSeXeDrF32x+WvfzmOjPiY9lg==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.8", - "for-each": "^0.3.3", - "gopd": "^1.2.0", - "has-proto": "^1.2.0", - "is-typed-array": "^1.1.14" - } - }, - "typed-array-byte-offset": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.4.tgz", - "integrity": "sha512-bTlAFB/FBYMcuX81gbL4OcpH5PmlFHqlCCpAl8AlEzMz5k53oNDvN8p1PNOWLEmI2x4orp3raOFB51tv9X+MFQ==", - "dev": true, - "peer": true, - "requires": { - "available-typed-arrays": "^1.0.7", - "call-bind": "^1.0.8", - "for-each": "^0.3.3", - "gopd": "^1.2.0", - "has-proto": "^1.2.0", - "is-typed-array": "^1.1.15", - "reflect.getprototypeof": "^1.0.9" - } - }, - "typed-array-length": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.7.tgz", - "integrity": "sha512-3KS2b+kL7fsuk/eJZ7EQdnEmQoaho/r6KUef7hxvltNA5DR8NAUM+8wJMbJyZ4G9/7i3v5zPBIMN5aybAh2/Jg==", - "dev": true, - "peer": true, - "requires": { - "call-bind": "^1.0.7", - "for-each": "^0.3.3", - "gopd": "^1.0.1", - "is-typed-array": "^1.1.13", - "possible-typed-array-names": "^1.0.0", - "reflect.getprototypeof": "^1.0.6" - } - }, - "typescript": { - "version": "5.8.3", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.8.3.tgz", - "integrity": "sha512-p1diW6TqL9L07nNxvRMM7hMMw4c5XOo/1ibL4aAIGmSAt9slTE1Xgw5KWuof2uTOvCg9BY7ZRi+GaF+7sfgPeQ==", - "dev": true, - "peer": true - }, - "unbox-primitive": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.1.0.tgz", - "integrity": "sha512-nWJ91DjeOkej/TA8pXQ3myruKpKEYgqvpw9lz4OPHj/NWFNluYrjbz9j01CJ8yKQd2g4jFoOkINCTW2I5LEEyw==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.3", - "has-bigints": "^1.0.2", - "has-symbols": "^1.1.0", - "which-boxed-primitive": "^1.1.1" - } - }, "uri-js": { "version": "4.4.1", "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", @@ -5776,71 +1642,6 @@ "isexe": "^2.0.0" } }, - "which-boxed-primitive": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/which-boxed-primitive/-/which-boxed-primitive-1.1.1.tgz", - "integrity": "sha512-TbX3mj8n0odCBFVlY8AxkqcHASw3L60jIuF8jFP78az3C2YhmGvqbHBpAjTRH2/xqYunrJ9g1jSyjCjpoWzIAA==", - "dev": true, - "peer": true, - "requires": { - "is-bigint": "^1.1.0", - "is-boolean-object": "^1.2.1", - "is-number-object": "^1.1.1", - "is-string": "^1.1.1", - "is-symbol": "^1.1.1" - } - }, - "which-builtin-type": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/which-builtin-type/-/which-builtin-type-1.2.1.tgz", - "integrity": "sha512-6iBczoX+kDQ7a3+YJBnh3T+KZRxM/iYNPXicqk66/Qfm1b93iu+yOImkg0zHbj5LNOcNv1TEADiZ0xa34B4q6Q==", - "dev": true, - "peer": true, - "requires": { - "call-bound": "^1.0.2", - "function.prototype.name": "^1.1.6", - "has-tostringtag": "^1.0.2", - "is-async-function": "^2.0.0", - "is-date-object": "^1.1.0", - "is-finalizationregistry": "^1.1.0", - "is-generator-function": "^1.0.10", - "is-regex": "^1.2.1", - "is-weakref": "^1.0.2", - "isarray": "^2.0.5", - "which-boxed-primitive": "^1.1.0", - "which-collection": "^1.0.2", - "which-typed-array": "^1.1.16" - } - }, - "which-collection": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.2.tgz", - "integrity": "sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw==", - "dev": true, - "peer": true, - "requires": { - "is-map": "^2.0.3", - "is-set": "^2.0.3", - "is-weakmap": "^2.0.2", - "is-weakset": "^2.0.3" - } - }, - "which-typed-array": { - "version": "1.1.19", - "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.19.tgz", - "integrity": "sha512-rEvr90Bck4WZt9HHFC4DJMsjvu7x+r6bImz0/BrbWb7A2djJ8hnZMrWnHo9F8ssv0OMErasDhftrfROTyqSDrw==", - "dev": true, - "peer": true, - "requires": { - "available-typed-arrays": "^1.0.7", - "call-bind": "^1.0.8", - "call-bound": "^1.0.4", - "for-each": "^0.3.5", - "get-proto": "^1.0.1", - "gopd": "^1.2.0", - "has-tostringtag": "^1.0.2" - } - }, "word-wrap": { "version": "1.2.5", "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", diff --git a/package.json b/package.json index d1560da..62b528b 100755 --- a/package.json +++ b/package.json @@ -29,7 +29,7 @@ "compatibility": "^4.0.0" }, "devDependencies": { - "eslint": "9.28.0", - "eslint-config-nodebb": "1.1.7" + "eslint": "10.0.3", + "eslint-config-nodebb": "^2.0.0" } } diff --git a/plugin.json b/plugin.json index 4d54462..e199564 100755 --- a/plugin.json +++ b/plugin.json @@ -6,19 +6,19 @@ "library": "library.js", "hooks": [ { "hook": "static:app.load", "method": "load" }, + { "hook": "filter:config.get", "method": "hooks.filter.configGet" }, { "hook": "filter:admin.header.build", "method": "addAdminNavigation" }, { "hook": "filter:composer.formatting", "method": "registerFormatting" }, { "hook": "static:privileges.categories.init", "method": "addPrivilege" }, { "hook": "filter:categories.copyPrivilegesFrom", "method": "copyPrivilegesFrom" }, { "hook": "filter:topic.post", "method": "hooks.filter.topicPost" }, - { "hook": "filter:parse.raw", "method": "hooks.filter.parseRaw", "priority": 1 }, + { "hook": "filter:topic.reply", "method": "hooks.filter.topicReply" }, { "hook": "filter:post.create", "method": "hooks.filter.postCreate" }, + { "hook": "filter:post.getFields", "method": "hooks.filter.postGetFields" }, { "hook": "filter:post.edit", "method": "hooks.filter.postEdit" }, { "hook": "action:post.delete", "method": "hooks.action.postDelete" }, { "hook": "action:post.restore", "method": "hooks.action.postRestore" }, - { "hook": "action:topic.delete", "method": "hooks.action.topicDelete" }, - { "hook": "action:topic.restore", "method": "hooks.action.topicRestore" }, { "hook": "filter:widgets.getWidgets", "method": "defineWidgets" }, { "hook": "filter:widget.render:poll", "method": "renderPollWidget" } @@ -40,6 +40,7 @@ "upgrades": [ "upgrades/give_categories_poll_create_privs_to_reg_users.js", "upgrades/convert_poll_id_voters_and_options_votes_to_sorted_sets.js", - "upgrades/fix_incorrect_poll_privilege.js" + "upgrades/fix_incorrect_poll_privilege.js", + "upgrades/poll-4.0-upgrade.js" ] } diff --git a/polls-todo.md b/polls-todo.md new file mode 100644 index 0000000..9b484b6 --- /dev/null +++ b/polls-todo.md @@ -0,0 +1,13 @@ +(done) ability to create polls in every post not just main post + +(done) ability to create more than 1 poll per post + +ability to edit polls + +ability to order polls in a post by drag and drop + +ability to order poll options by drag and drop + +TODO: + save/restore drafts + (done) composer formatting toolbar badge number to show how many polls there are \ No newline at end of file diff --git a/public/js/admin.js b/public/js/admin.js index a91e389..fad94fd 100755 --- a/public/js/admin.js +++ b/public/js/admin.js @@ -1,44 +1,15 @@ 'use strict'; define('admin/plugins/poll', ['settings'], function (Settings) { - var wrapper; - - var ACP = {}; + const ACP = {}; ACP.init = function () { - wrapper = $('.poll-settings'); - - Settings.sync('poll', wrapper); + Settings.load('poll', $('.poll-settings')); $('#save').on('click', function () { - save(); - }); - - $('#reset').click(function () { - reset(); + Settings.save('poll', $('.poll-settings')); }); }; - function save() { - Settings.persist('poll', wrapper, function () { - socket.emit('admin.plugins.poll.sync'); - }); - } - - function reset() { - bootbox.confirm('Are you sure you wish to reset the settings?', function (sure) { - if (sure) { - socket.emit('admin.plugins.poll.getDefaults', null, function (err, data) { - if (err) { - console.error(err); - } - Settings.set('poll', data, wrapper, function () { - socket.emit('admin.plugins.poll.sync'); - }); - }); - } - }); - } - return ACP; }); diff --git a/public/js/poll/creator.js b/public/js/poll/creator.js index 68788ac..6cc6112 100755 --- a/public/js/poll/creator.js +++ b/public/js/poll/creator.js @@ -1,203 +1,211 @@ 'use strict'; (function (Poll) { - var Creator = {}; + const Creator = {}; function init() { $(window).on('action:composer.enhanced', initComposer); - - $(window).on('action:redactor.load', initRedactor); - - $(window).on('action:composer.loaded', function (ev, data) { - if ($.Redactor) { - if (data.composerData.isMain && $.Redactor.opts.plugins.indexOf('poll') === -1) { - $.Redactor.opts.plugins.push('poll'); - } else if (!data.composerData.isMain && $.Redactor.opts.plugins.indexOf('poll') !== -1) { - $.Redactor.opts.plugins.splice($.Redactor.opts.plugins.indexOf('poll'), 1); - } - } - }); } function initComposer() { - require(['composer', 'composer/formatting', 'composer/controls'], function (composer, formatting, controls) { - if (formatting && controls) { - formatting.addButtonDispatch('poll', function (textarea) { - composerBtnHandle(composer, textarea); + require(['composer', 'composer/formatting'], function (composer, formatting) { + if (formatting) { + formatting.addButtonDispatch('poll', function (/* textarea */) { + onPollButtonClicked(composer); }); } }); } - function initRedactor() { - $.Redactor.prototype.poll = function () { - return { - init: function () { - var self = this; - - // require translator as such because it was undefined without it - require(['translator'], function (translator) { - translator.translate('[[poll:creator_title]]', function (translated) { - var button = self.button.add('poll', translated); - self.button.setIcon(button, ''); - self.button.addCallback(button, self.poll.onClick); - }); - }); - }, - onClick: function () { - var self = this; - var code = this.code.get(); - require(['composer'], function (composer) { - composerBtnHandle(composer, { - value: code, - redactor: function (code) { - self.code.set(code); - }, - }); - }); - }, - }; - }; + async function onPollButtonClicked(composer) { + const postData = composer.posts[composer.active]; + if (!postData) { + return showAlert('error', '[[poll:error.invalid-post]]'); + } + if (parseInt(postData.cid, 10) === 0) { + return showAlert('error', '[[error:category-not-selected]]'); + } + + const canCreate = await socket.emit('plugins.poll.canCreate', { + cid: postData.cid, + tid: postData.tid, + }); + if (!canCreate) { + return showAlert('error', '[[error:no-privileges]]'); + } + postData.polls = postData.polls || []; + await openManageModal({ postData, uuid: composer.active }); } - function composerBtnHandle(composer, textarea) { - require(['composer/controls', 'alerts'], function (controls, alerts) { - var post = composer.posts[composer.active]; - if (!post || !post.isMain || (isNaN(parseInt(post.cid, 10)) && isNaN(parseInt(post.pid, 10)))) { - return alerts.error('[[poll:error.not_main]]'); - } - if (parseInt(post.cid, 10) === 0) { - return alerts.error('[[error:category-not-selected]]'); - } + async function openManageModal(payload) { + const { postData, modal, uuid } = payload; + const html = await app.parseAndTranslate('poll/manage', { post: postData }); + + if (modal) { + modal.find('.bootbox-body').html(html[0].outerHTML); + handleSort({ modal, polls: postData.polls }); + } else { + const modal = bootbox.dialog({ + title: '[[poll:manage-polls]]', + message: html, + className: 'poll-manage', + buttons: { + add: { + label: ' [[poll:add-poll]]', + className: 'btn-success', + callback: () => { + Creator.show({ + // temp id, will be replaced in the backend before saving + // used to identify the poll in the manage modal after creation + // so we can edit/delete + pollId: Date.now(), + }, config.poll).then((newPoll) => { + if (newPoll) { + postData.polls.push(newPoll); + } + openManageModal({ ...payload, modal }); + updateComposerBadgeCount(uuid, postData.polls.length); + }); + return false; + }, + }, + close: { + label: '[[global:close]]', + className: 'btn-primary', + }, + }, + }); + handleDelete({ ...payload, modal }); + handleEdit({ ...payload, modal }); + handleSort({ modal, polls: postData.polls }); + } + } - socket.emit('plugins.poll.canCreate', { - cid: post.cid, - pid: post.pid, - }, function (err, canCreate) { - if (err) { - return alerts.error(err); + function handleDelete(payload) { + const { uuid, modal, postData } = payload; + modal.on('click', 'button[data-action="remove"]', function () { + const clickedPollId = $(this).attr('data-poll-id'); + bootbox.confirm('[[poll:confirm-remove]]', (ok) => { + if (!ok) { + return; } - if (!canCreate) { - return alerts.error('[[error:no-privileges]]'); + const poll = postData?.polls.find(poll => String(poll.pollId) === clickedPollId); + if (poll) { + postData.polls = postData.polls.filter(poll => String(poll.pollId) !== clickedPollId); + openManageModal(payload); + updateComposerBadgeCount(uuid, postData.polls.length); } - socket.emit('plugins.poll.getConfig', null, function (err, config) { - if (err) { - return alerts.error(err); - } - - var poll = {}; + }); + }); + }; - // If there's already a poll in the post, serialize it for editing - if (Poll.serializer.canSerialize(textarea.value)) { - poll = Poll.serializer.serialize(textarea.value, config); + function handleEdit(payload) { + const { modal, postData } = payload; + modal.on('click', 'button[data-action="edit"]', async function () { + const clickedPollId = $(this).attr('data-poll-id'); + const poll = postData?.polls.find(poll => String(poll.pollId) === clickedPollId); + if (poll) { + const editedPoll = await Creator.show(poll, payload.config); + if (editedPoll) { + postData.polls = postData.polls.map( + p => String(p.pollId) === String(editedPoll.pollId) ? editedPoll : p + ); + openManageModal(payload); + } + } + }); + } - if (poll.settings.end === 0) { - delete poll.settings.end; - } else { - poll.settings.end = parseInt(poll.settings.end, 10); - } + function handleSort({ modal, polls }) { + if (Array.isArray(polls) && polls.length > 1) { + const selectorEl = modal.find('.topic-poll-modal'); + selectorEl.sortable({ + handle: '[component="sort/handle"]', + axis: 'y', + zIndex: 9999, + items: '[component="post/poll/item"]', + }); + selectorEl.on('sortupdate', function () { + if (!polls) return; + const newOrder = []; + selectorEl.find('[component="post/poll/item"]').each(function () { + const pollId = $(this).attr('data-poll-id'); + const poll = polls.find(p => String(p.pollId) === pollId); + if (poll) { + newOrder.push(poll); } - - Creator.show(poll, config, function (data) { - // Anything invalid will be discarded by the serializer - var markup = Poll.serializer.deserialize(data, config); - - // Remove any existing poll markup - textarea.value = Poll.serializer.removeMarkup(textarea.value); - - // Insert the poll markup at the bottom - if (textarea.value.charAt(textarea.value.length - 1) !== '\n') { - markup = '\n' + markup; - } - - if ($.Redactor) { - textarea.redactor(textarea.value + '

' + markup + '

'); - } else { - controls.insertIntoTextarea(textarea, markup); - } - }); }); + // Mutate polls array in place + polls.length = 0; + Array.prototype.push.apply(polls, newOrder); }); - }); - } - - Creator.show = function (poll, config, callback) { - if (poll.hasOwnProperty('info')) { - return Poll.alertError('Editing not implemented'); } + }; - require(['bootbox'], function (bootbox) { - app.parseAndTranslate('poll/creator', { poll: poll, config: config, isRedactor: !!$.Redactor }, function (html) { - // Initialise modal - var modal = bootbox.dialog({ - title: '[[poll:creator_title]]', - message: html, - className: 'poll-creator', - buttons: { - cancel: { - label: '[[modules:bootbox.cancel]]', - callback: function () { - return true; + Creator.show = function (poll, config) { + return new Promise((resolve) => { + require(['bootbox'], function (bootbox) { + app.parseAndTranslate('poll/creator', { poll, config }, function (html) { + const modal = bootbox.dialog({ + title: '[[poll:creator_title]]', + message: html, + className: 'poll-creator', + buttons: { + cancel: { + label: '[[global:close]]', + callback: function () { + resolve(null); + }, }, - }, - save: { - label: '[[modules:bootbox.confirm]]', - callback: function (e) { - clearErrors(); - var form = $(e.currentTarget).parents('.bootbox').find('#pollCreator'); - var obj = serializeObjectFromForm(form); - - // Let's be nice and at least show an error if there are no options - obj.options.filter(function (obj) { - return obj.length; - }); - - if (obj.options.length === 0) { - return error('[[poll:error.no_options]]'); - } - - if (obj.settings.end) { - obj.settings.end = new Date(obj.settings.end).getTime(); - } - callback(obj); - return true; + save: { + label: '[[modules:bootbox.confirm]]', + callback: function () { + clearErrors(); + const form = modal.find('#pollCreator'); + const newPoll = serializeObjectFromForm(form); + + // Let's be nice and at least show an error if there are no options + if (newPoll.options.length === 0) { + error('[[poll:error.no_options]]'); + return false; + } + + resolve(newPoll); + return true; + }, }, }, - }, - }); + }); - // Add option adder - modal.find('#pollAddOption') - .off('click') - .on('click', function (e) { - var el = $(e.currentTarget); - var prevOption = el.prev(); - - if (config.limits.maxOptions <= el.prevAll('input').length) { - clearErrors(); - require(['translator'], function (translator) { - translator.translate('[[poll:error.max_options]]', function (text) { - error(text.replace('%d', config.limits.maxOptions)); - }); + // Add option adder + modal.find('#pollAddOption') + .off('click') + .on('click', async function (e) { + const el = $(e.currentTarget); + if (config.maxOptions <= el.prevAll('input').length) { + clearErrors(); + error(`[[poll:error.max_options, ${config.maxOptions}]]`); + return false; + } + const id = Date.now(); + const newOptionInput = await app.parseAndTranslate('poll/option-input', { + id: id, + title: '', }); - return false; - } - - if (prevOption.val().length !== 0) { - prevOption.clone().val('').insertBefore(el).focus(); - } - }); + newOptionInput.insertBefore(el); + modal.find(`[data-option-id="${id}"] input`).focus(); + }); + }); }); }); }; - function error(message) { - var errorBox = $('#pollErrorBox'); - + async function error(message) { + const errorBox = $('#pollErrorBox'); + const translator = await app.require('translator'); + const msgTranslated = await translator.translate(message); errorBox.removeClass('hidden'); - errorBox.append(message + '
'); - - return false; + errorBox.append(msgTranslated + '
'); } function clearErrors() { @@ -205,21 +213,46 @@ } function serializeObjectFromForm(form) { - var obj = form.serializeObject(); - var result = { - options: obj.options, - settings: { - title: obj['settings.title'], - maxvotes: obj['settings.maxvotes'], - disallowVoteUpdate: obj['settings.disallowVoteUpdate'] === 'on' ? 'true' : 'false', - allowAnonVoting: obj['settings.allowAnonVoting'] === 'on' ? 'true' : 'false', - end: obj['settings.end'], - }, + const obj = form.serializeObject(); + obj.options = obj.options || []; + obj.id = obj.id || []; + + const options = obj.options.map((opt, index) => ({ + id: obj.id[index], + title: opt, + })).filter(option => option.title?.length); + + const result = { + pollId: obj.pollId, + options: options, + title: obj.title, + maximumVotesPerUser: obj.maximumVotesPerUser, + disallowVoteUpdate: obj.disallowVoteUpdate === 'on' ? 1 : 0, + allowAnonVoting: obj.allowAnonVoting === 'on' ? 1 : 0, + end: obj.end ? new Date(obj.end).getTime() : 0, }; return result; } + function showAlert(type, message) { + require(['alerts'], function (alerts) { + alerts[type](message); + }); + } + + function updateComposerBadgeCount(uuid, count) { + const postContainer = $(`[component="composer"][data-uuid="${uuid}"]`); + require(['composer'], (composer) => { + composer.updateFormattingBtnBadgeCount( + uuid, + postContainer, + 'poll', + count + ); + }); + } + Poll.creator = Creator; init(); diff --git a/public/js/poll/main.js b/public/js/poll/main.js index cc8f03b..443634f 100755 --- a/public/js/poll/main.js +++ b/public/js/poll/main.js @@ -4,26 +4,31 @@ window.Poll = {}; -(function () { +$(document).ready(function () { window.Poll.alertError = function (message) { require(['alerts'], function (alerts) { alerts.error(message); }); }; - require('poll/serializer')(window.utils); + require(['hooks'], function (hooks) { + // add polls to data that is sent to server on composer submit + hooks.on('filter:composer.submit', function (hookData) { + hookData.composerData.polls = hookData.postData.polls || []; + }); + }); $(window).on('action:topic.loading', function () { - if (ajaxify.data.posts.length > 0 && ajaxify.data.posts[0].hasOwnProperty('pollId')) { - getPollByPost(ajaxify.data.posts[0]); + if (ajaxify.data.posts.length > 0) { + ajaxify.data.posts.forEach((post) => { + getPollByPost(post); + }); } }); $(window).on('action:posts.loaded', function (ev, data) { data.posts.forEach((post) => { - if (post.hasOwnProperty('pollId')) { - getPollByPost(post); - } + getPollByPost(post); }); }); @@ -35,7 +40,7 @@ window.Poll = {}; $(window).on('action:ajaxify.end', function () { $('[data-widget-poll-id]').each(function () { - getPoll($(this).attr('data-widget-poll-id'), $(this)); + getPolls([$(this).attr('data-widget-poll-id')], $(this)); }); }); @@ -44,25 +49,34 @@ window.Poll = {}; }); function getPollByPost(post) { - const postEl = $('[component="post"][data-pid="' + post.pid + '"]'); - if (postEl.length && post.pollId) { - getPoll(post.pollId, postEl.find('[component="post/content"]')); + if (!post || !post.hasOwnProperty('pollIds') || !post.pollIds) return; + let pollIds; + try { + pollIds = JSON.parse(post.pollIds || '[]'); + } catch (err) { + pollIds = []; + } + if (pollIds.length) { + const postEl = $(`[component="post"][data-pid="${post.pid}"]`); + getPolls(pollIds, postEl.find('[component="post/content"]')); } } - function getPoll(pollId, container) { - pollId = parseInt(pollId, 10); - if (!isNaN(pollId)) { - if (!socket.connected) { - socket.connect(); + function getPolls(pollIds, container) { + const validPollIds = pollIds.map(id => parseInt(id, 10)).filter(id => !isNaN(id)); + if (!socket.connected) { + socket.connect(); + } + + socket.emit('plugins.poll.get', { pollIds: validPollIds.reverse() }, function (err, pollDataArr) { + if (err) { + return Poll.alertError(err.message); } - socket.emit('plugins.poll.get', { pollId }, function (err, pollData) { - if (err) { - return Poll.alertError(err.message); - } + + pollDataArr.forEach((pollData) => { pollData.container = container; Poll.view.load(pollData); }); - } + }); } -}()); +}); diff --git a/public/js/poll/serializer.js b/public/js/poll/serializer.js deleted file mode 100644 index 3d65a9a..0000000 --- a/public/js/poll/serializer.js +++ /dev/null @@ -1,173 +0,0 @@ -'use strict'; - -module.exports = function (utils) { - var Serializer = {}; - - var pollRegex = /(?:(?:\[poll(?.*?)\])(?:\\n|\n|
)(?(?:-.+?(?:\\n|\n|
))+)(?:\[\/poll\]))/g; - var settingsRegex = /(?.+?)=(?:"|"|\)(?.+?)(?:"|"|\)/g; - var settingsValidators = { - title: { - test: function (value) { - return value.length > 0; - }, - parse: function (value) { - return utils.stripHTMLTags(value).trim(); - }, - }, - maxvotes: { - test: function (value) { - return !isNaN(value); - }, - parse: function (value) { - return parseInt(value, 10); - }, - }, - disallowVoteUpdate: { - test: function (value) { - return /true|false/.test(value); - }, - parse: function (value) { - return value === 'true' || value === true ? 1 : 0; - }, - }, - allowAnonVoting: { - test: function (value) { - return /true|false/.test(value); - }, - parse: function (value) { - return value === 'true' || value === true ? 1 : 0; - }, - }, - end: { - test: function (value) { - return (!isNaN(value) && parseInt(value, 10) > Date.now()); - }, - parse: function (value) { - return parseInt(value, 10); - }, - }, - }; - - Serializer.canSerialize = function (post) { - pollRegex.lastIndex = 0; - return pollRegex.test(post); - }; - - Serializer.removeMarkup = function (content, replace) { - return content.replace(pollRegex, replace || ''); - }; - - Serializer.hasMarkup = function (content) { - pollRegex.lastIndex = 0; - const has = pollRegex.test(content); - return has; - }; - - Serializer.serialize = function (post, config) { - pollRegex.lastIndex = 0; - var match = pollRegex.exec(post); - - if (match === null) { - return null; - } - - return { - options: serializeOptions(match.groups.content, config), - settings: serializeSettings(match.groups.settings, config), - }; - }; - - Serializer.deserialize = function (poll, config) { - var options = deserializeOptions(poll.options, config); - var settings = deserializeSettings(poll.settings, config); - - return '[poll' + settings + ']\n' + options + '\n[/poll]'; - }; - - function serializeOptions(raw, config) { - // Depending on composer, the line breaks can either be \n or
so handle both - var pollOptions = []; - var rawOptions = raw.split(/(?:\\n|\n|
)/); - rawOptions.map(raw => utils.stripHTMLTags(raw)); - var maxOptions = parseInt(config.limits.maxOptions, 10); - - rawOptions.forEach(function (option) { - if (option.length) { - option = option.split('-').slice(1).join('-').trim(); - - if (option.length) { - pollOptions.push(option); - } - } - }); - - if (pollOptions.length > maxOptions) { - pollOptions = pollOptions.slice(0, maxOptions - 1); - } - - return pollOptions; - } - - function deserializeOptions(options, config) { - var maxOptions = config.limits.maxOptions; - - options = options.map(function (option) { - return utils.stripHTMLTags(option).trim(); - }).filter(function (option) { - return option.length; - }); - - if (options.length > maxOptions) { - options = options.slice(0, maxOptions - 1); - } - - return options.length ? '- ' + options.join('\n- ') : ''; - } - - function serializeSettings(raw, config) { - var settings = {}; - - Object.keys(config.defaults).forEach(function (key) { - settings[key] = config.defaults[key]; - }); - - const stripped = utils.stripHTMLTags(raw).replace(/\\/g, '\'); - let match; - while ((match = settingsRegex.exec(stripped)) !== null) { - var key = match.groups.key.trim(); - var value = match.groups.value.trim(); - - if (key.length && value.length && settingsValidators.hasOwnProperty(key)) { - if (settingsValidators[key].test(value)) { - settings[key] = settingsValidators[key].parse(value); - } - } - } - - return settings; - } - - function deserializeSettings(settings, config) { - var deserialized = ''; - - for (var k of Object.keys(settings)) { - if (settings.hasOwnProperty(k) && config.defaults.hasOwnProperty(k)) { - var key = utils.stripHTMLTags(k).trim(); - var value = utils.stripHTMLTags(settings[k]).trim(); - - if (key.length && value.length && settingsValidators.hasOwnProperty(key)) { - if (settingsValidators[key].test(value)) { - deserialized += ' ' + key + '="' + value + '"'; - } - } - } - } - - return deserialized; - } - - if (typeof window !== 'undefined') { - window.Poll.serializer = Serializer; - } - return Serializer; -}; diff --git a/public/js/poll/view.js b/public/js/poll/view.js index 22cf0bd..73f93cc 100644 --- a/public/js/poll/view.js +++ b/public/js/poll/view.js @@ -2,13 +2,11 @@ (function (Poll) { function vote(view, options) { - var form = view.dom.votingPanel.find('form'); - var votes = form.serializeArray().map(function (option) { - return parseInt(option.value, 10); - }); + const form = view.dom.votingPanel.find('form'); + const votes = form.serializeArray().map(option => parseInt(option.value, 10)); if (votes.length > 0) { - var voteData = { + const voteData = { pollId: view.pollData.info.pollId, options: votes, voteAnon: options.voteAnon, @@ -28,11 +26,11 @@ } } - var Actions = [ + const Actions = [ { // Voting register: function (view) { - var self = this; + const self = this; view.dom.voteButton.off('click').on('click', function () { self.handle(view); }); @@ -46,7 +44,7 @@ { // vote anon register: function (view) { - var self = this; + const self = this; view.dom.voteAnonButton.off('click').on('click', function () { self.handle(view); }); @@ -60,19 +58,19 @@ { // update voting register: function (view) { - var self = this; + const self = this; view.dom.updateVoteButton.off('click').on('click', function () { self.handle(view); }); }, handle: function (view) { - var form = view.dom.votingPanel.find('form'); - var votes = form.serializeArray().map(function (option) { + const form = view.dom.votingPanel.find('form'); + const votes = form.serializeArray().map(function (option) { return parseInt(option.value, 10); }); if (votes.length > 0) { - var voteData = { + const voteData = { pollId: view.pollData.info.pollId, options: votes, }; @@ -88,13 +86,13 @@ { // Remove vote register: function (view) { - var self = this; + const self = this; view.dom.removeVoteButton.off('click').on('click', function () { self.handle(view); }); }, handle: function (view) { - var voteData = { pollId: view.pollData.info.pollId }; + const voteData = { pollId: view.pollData.info.pollId }; socket.emit('plugins.poll.removeVote', voteData, function (err) { if (err) { @@ -107,7 +105,7 @@ { // Results button register: function (view) { - var self = this; + const self = this; view.dom.resultsPanelButton.off('click').on('click', function () { self.handle(view); }); @@ -119,7 +117,7 @@ { // To Voting button register: function (view) { - var self = this; + const self = this; view.dom.votingPanelButton.off('click').on('click', function () { self.handle(view); }); @@ -131,13 +129,13 @@ { // Option details register: function (view) { - var self = this; + const self = this; view.dom.resultsPanel.off('click').on('click', '.poll-result-votecount', function (e) { self.handle(view, e); }); }, handle: function (view, e) { - var optionId = $(e.currentTarget).parents('[data-poll-option-id]').data('poll-option-id'); + const optionId = $(e.currentTarget).parents('[data-poll-option-id]').data('poll-option-id'); socket.emit('plugins.poll.getOptionDetails', { pollId: view.pollData.info.pollId, @@ -154,7 +152,7 @@ { // Editing register: function (view) { - var self = this; + const self = this; view.dom.editButton.off('click').on('click', function () { self.handle(view); }); @@ -170,12 +168,12 @@ }, ]; - var View = function (pollData) { + const View = function (pollData) { this.pollData = pollData; }; View.prototype.load = function () { - var self = this; + const self = this; if (!self.pollData.container.length) { return; } @@ -219,7 +217,7 @@ }; View.prototype.voteUpdateAllowed = function () { - return parseInt(this.pollData.settings.disallowVoteUpdate, 10) !== 1; + return parseInt(this.pollData.disallowVoteUpdate, 10) !== 1; }; View.prototype.pollEndedOrDeleted = function () { @@ -252,7 +250,7 @@ this.pollEndedOrDeleted(); this.pollData.options.forEach(function (option) { - var el = this.dom.resultsPanel.find('[data-poll-option-id=' + option.id + ']'); + const el = this.dom.resultsPanel.find('[data-poll-option-id=' + option.id + ']'); el.find('.poll-result-votecount span').translateText(`[[poll:x-votes, ${option.voteCount}]]`); el.find('.poll-result-progressbar').css('width', option.percentage + '%') .find('span.percent').text(option.percentage); @@ -264,7 +262,7 @@ }; View.prototype.showMessage = function (title, content) { - var self = this; + const self = this; app.parseAndTranslate('poll/view/messages', { title: title, content: content }, function (html) { self.dom.messages.html(html).removeClass('hidden'); @@ -279,6 +277,7 @@ require(['bootbox'], function (bootbox) { app.parseAndTranslate('poll/view/details', details, function (html) { bootbox.dialog({ + title: `[[poll:x-users-voted-for-this-option, ${details.voteCount}]]`, message: html, backdrop: true, buttons: { @@ -292,7 +291,7 @@ }; View.prototype.fillVotingForm = function () { - var self = this; + const self = this; this.resetVotingForm(); if (this.pollData.vote && this.pollData.vote.options) { this.pollData.vote.options.forEach(function (id) { @@ -318,7 +317,7 @@ } } else { this.showVoteButton(); - if (this.pollData.settings.allowAnonVoting) { + if (this.pollData.allowAnonVoting) { this.showVoteAnonButton(); } } @@ -401,13 +400,13 @@ Poll.view = { polls: {}, load: function (pollData) { - var view = new View(pollData); + const view = new View(pollData); this.polls[pollData.info.pollId] = view; view.load(); }, update: function (pollData, uid) { - var pollId = pollData.info.pollId; + const pollId = pollData.info.pollId; if (this.polls.hasOwnProperty(pollId)) { this.polls[pollId].update(pollData, uid); } diff --git a/schema.md b/schema.md new file mode 100644 index 0000000..ef3090e --- /dev/null +++ b/schema.md @@ -0,0 +1,24 @@ + +post:100 { + ... + pollIds: ["200", "201", ...] +} + +poll:200 { + "_key" : "poll:200", + "deleted" : 0, + "end" : 0, // or unix timestamp when voting will close + "pid" : 100, + "pollId" : 200, + "timestamp" : 1772893625284, + "uid" : 1, + "options": [ + { id: "300", title: "option 1" }, { id: "301", title: "option 2" }, { id: "302", title: "option 3" } + ] +} + +poll:123:voters // uids of voters zset + +poll:123:options:300:votes { //zset of uids who voted for this option + score: "", value: "1" +} \ No newline at end of file diff --git a/templates/poll/creator.tpl b/templates/poll/creator.tpl index 7b3308c..394c000 100755 --- a/templates/poll/creator.tpl +++ b/templates/poll/creator.tpl @@ -1,45 +1,45 @@ -
+ + +
- +
-
+
- - - - - - - - + {{{ if poll.options.length }}} + {{{ each poll.options }}} + + {{{ end }}} + {{{ end }}} +

- + +

[[poll:info_choices]]

- +
- +
- +

[[poll:auto_end_help]]

diff --git a/templates/poll/manage.tpl b/templates/poll/manage.tpl new file mode 100644 index 0000000..63de4de --- /dev/null +++ b/templates/poll/manage.tpl @@ -0,0 +1,18 @@ +
+ {{{ if !post.polls.length }}} +
[[poll:no-polls]]
+ {{{ end }}} + {{{ each post.polls }}} +
+
+ +
{./title}
+
+ +
+ + +
+
+ {{{ end }}} +
diff --git a/templates/poll/option-input.tpl b/templates/poll/option-input.tpl new file mode 100644 index 0000000..833d33b --- /dev/null +++ b/templates/poll/option-input.tpl @@ -0,0 +1,5 @@ +
+ + + +
\ No newline at end of file diff --git a/templates/poll/view.tpl b/templates/poll/view.tpl index 3b55b1b..b93f4ab 100644 --- a/templates/poll/view.tpl +++ b/templates/poll/view.tpl @@ -1,7 +1,7 @@
-
{poll.settings.title}
+
{poll.info.title}
- {buildAvatar(votes, "24px", true)} - - +
+ {{{ each votes }}} + + {buildAvatar(votes, "24px", true)} + + {{{ end }}} +
diff --git a/upgrades/poll-4.0-upgrade.js b/upgrades/poll-4.0-upgrade.js new file mode 100644 index 0000000..6e94abd --- /dev/null +++ b/upgrades/poll-4.0-upgrade.js @@ -0,0 +1,125 @@ +'use strict'; + +const db = require.main.require('./src/database'); +const batch = require.main.require('./src/batch'); + +module.exports = { + name: 'Convert data structures for v4.0', + timestamp: Date.UTC(2026, 2, 7), + method: async function () { + const { progress } = this; + + await convertSettings(); + + await db.delete('polls:scheduled'); + + const pollIds = [...new Set(await db.getListRange('polls', 0, -1))]; + progress.total = pollIds.length; + + await batch.processArray(pollIds, async (pollIds) => { + let polls = await db.getObjects(pollIds.map(pollId => `poll:${pollId}`)); + + await moveSettingsToPollHash(pollIds); + + await addPollsToSortedSet(polls, pollIds); + + await savePollOptionsInPollHash(pollIds); + + await updatePollIdsInPosts(polls, pollIds); + + progress.incr(pollIds.length); + }, { + batch: 500, + }); + await db.delete('polls'); + }, +}; + +async function convertSettings() { + const settings = await db.getObject('settings:poll'); + if (settings) { + try { + const parsed = JSON.parse(settings._); + const newSettings = {}; + if (parsed?.toggles?.allowAnon) { + newSettings.allowGuestsToViewResults = parsed.toggles.allowAnon; + } + if (parsed?.limits?.maxOptions) { + newSettings.maxOptions = parsed.limits.maxOptions; + } + if (parsed?.defaults?.title) { + newSettings.defaultTitle = parsed.defaults.title; + } + if (parsed?.defaults?.maxvotes) { + newSettings.maximumVotesPerUser = parsed.defaults.title; + } + await db.setObject('settings:poll', newSettings); + } catch (err) { + console.info('[Poll Plugin] Error parsing settings, resetting to defaults'); + } + } +} + +async function moveSettingsToPollHash(pollIds) { + const pollSettings = await db.getObjects(pollIds.map(pollId => `poll:${pollId}:settings`)); + const bulkSet = []; + pollSettings.forEach((s, i) => { + const pollId = pollIds[i]; + if (s && pollId) { + bulkSet.push([`poll:${pollId}`, { + title: Object.hasOwn(s, 'title') ? s.title : 'Poll', + end: Object.hasOwn(s, 'end') ? s.end : 0, + disallowVoteUpdate: Object.hasOwn(s, 'disallowVoteUpdate') ? s.disallowVoteUpdate : 0, + allowAnonVoting: Object.hasOwn(s, 'allowAnonVoting') ? s.allowAnonVoting : 0, + maximumVotesPeruser: Object.hasOwn(s, 'maxvotes') ? s.maxvotes : 0, + }]); + } + }); + await db.setObjectBulk(bulkSet); + await db.deleteAll(pollIds.map(pollId => `poll:${pollId}:settings`)); +} + +async function addPollsToSortedSet(polls, pollIds) { + const bulkAdd = []; + polls.forEach((poll, index) => { + if (poll) { + bulkAdd.push([`polls:createtime`, poll.timestamp || 0, pollIds[index]]); + } + }); + await db.sortedSetAddBulk(bulkAdd); +} + +async function savePollOptionsInPollHash(pollIds) { + const pollOptions = await db.getSetsMembers(pollIds.map(pollId => `poll:${pollId}:options`)); + const bulkSet = []; + await Promise.all(pollOptions.map(async (options, index) => { + const pollId = pollIds[index]; + if (pollId && options?.length) { + const keys = options.map((option, i) => `poll:${pollId}:options:${options[i]}`) + const optionData = await db.getObjects(keys); + bulkSet.push([ + `poll:${pollId}`, + { options: JSON.stringify(optionData.map(o => ({ id: o.id, title: o.title }))) }, + ]); + } + })); + + await db.setObjectBulk(bulkSet); + await db.deleteAll(pollIds.map(pollId => `poll:${pollId}:options`)) +} + +async function updatePollIdsInPosts(polls, pollIds) { + const pids = polls.map(poll => poll && poll.pid); + const bulkSet = []; + polls.forEach((poll, index) => { + if (poll && poll.pid) { + bulkSet.push([ + `post:${poll.pid}`, + { pollIds: JSON.stringify([pollIds[index]]) } + ]); + } + }); + + await db.setObjectBulk(bulkSet); + await db.deleteObjectFields(pids.map(pid => `post:${pid}`), ['pollId']); +} \ No newline at end of file From 68cb4099569a686d5e37c23a0e512cc5a735091b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Sun, 8 Mar 2026 00:48:09 -0500 Subject: [PATCH 02/14] composer.push for edit --- lib/hooks.js | 19 +++++++++++++++++++ lib/poll.js | 2 +- lib/sockets.js | 10 +++++++--- library.js | 10 ---------- plugin.json | 3 ++- public/js/poll/creator.js | 15 +++++++++------ 6 files changed, 38 insertions(+), 21 deletions(-) diff --git a/lib/hooks.js b/lib/hooks.js index 73487a6..64e9b53 100755 --- a/lib/hooks.js +++ b/lib/hooks.js @@ -19,6 +19,25 @@ Hooks.filter.configGet = async function (config) { return config; }; +Hooks.filter.registerFormatting = function (payload) { + payload.options.push({ + name: 'poll', + className: `fa ${Config.plugin.icon}`, + title: '[[poll:creator_title]]', + badge: true, + }); + return payload; +}; + +Hooks.filter.composerPush = async function (hookData) { + if (hookData.pid) { + const pollIds = await Poll.getPollIdsByPid(hookData.pid); + const polls = await Promise.all(pollIds.map(pollId => Poll.get(pollId, 0))); + hookData.polls = polls.map(p => p && p.info).filter(Boolean); + } + return hookData; +}; + Hooks.filter.postCreate = async function (hookData) { // post is going to be saved to db, data is what is submitted by user const { post, data } = hookData; diff --git a/lib/poll.js b/lib/poll.js index 95fdde2..8b5ef9c 100755 --- a/lib/poll.js +++ b/lib/poll.js @@ -47,7 +47,7 @@ Poll.add = async function (postData, polls) { ]); }; -Poll.get = async function (pollId, uid, withVotes) { +Poll.get = async function (pollId, uid, withVotes = false) { const [info, hasVoted, vote] = await Promise.all([ Poll.getInfo(pollId, withVotes), uid ? Vote.hasUidVoted(uid, pollId) : false, diff --git a/lib/sockets.js b/lib/sockets.js index 18e3a2c..0c27649 100755 --- a/lib/sockets.js +++ b/lib/sockets.js @@ -2,6 +2,8 @@ const _ = require.main.require('lodash'); const db = require.main.require('./src/database'); +const posts = require.main.require('./src/posts'); +const topics = require.main.require('./src/topics'); const NodeBB = require('./nodebb'); const Config = require('./config'); @@ -170,9 +172,11 @@ Sockets.canCreate = async function (socket, data) { if (!socket.uid || !data) { throw new Error('Invalid request'); } - let { cid } = data; - if (!cid) { - cid = await NodeBB.Topics.getTopicField(data.tid, 'cid'); + let { cid, tid, pid } = data; + if (pid) { + cid = await posts.getCidByPid(pid); + } else if (tid) { + cid = await topics.getTopicField(tid, 'cid'); } return await checkPrivs(cid, socket.uid); }; diff --git a/library.js b/library.js index 4f8b6f7..4f30d83 100755 --- a/library.js +++ b/library.js @@ -33,16 +33,6 @@ Plugin.addAdminNavigation = function (adminHeader) { return adminHeader; }; -Plugin.registerFormatting = function (payload) { - payload.options.push({ - name: 'poll', - className: `fa ${Config.plugin.icon}`, - title: '[[poll:creator_title]]', - badge: true, - }); - return payload; -}; - Plugin.addPrivilege = function (hookData) { hookData.privileges.set( 'poll:create', { label: '[[poll:admin.create-poll]]' }, diff --git a/plugin.json b/plugin.json index e199564..51b7cc9 100755 --- a/plugin.json +++ b/plugin.json @@ -8,7 +8,8 @@ { "hook": "static:app.load", "method": "load" }, { "hook": "filter:config.get", "method": "hooks.filter.configGet" }, { "hook": "filter:admin.header.build", "method": "addAdminNavigation" }, - { "hook": "filter:composer.formatting", "method": "registerFormatting" }, + { "hook": "filter:composer.formatting", "method": "hooks.filter.registerFormatting" }, + { "hook": "filter:composer.push", "method": "hooks.filter.composerPush" }, { "hook": "static:privileges.categories.init", "method": "addPrivilege" }, { "hook": "filter:categories.copyPrivilegesFrom", "method": "copyPrivilegesFrom" }, { "hook": "filter:topic.post", "method": "hooks.filter.topicPost" }, diff --git a/public/js/poll/creator.js b/public/js/poll/creator.js index 6cc6112..81b3e9b 100755 --- a/public/js/poll/creator.js +++ b/public/js/poll/creator.js @@ -7,12 +7,14 @@ $(window).on('action:composer.enhanced', initComposer); } - function initComposer() { + function initComposer(ev, { postContainer, postData }) { require(['composer', 'composer/formatting'], function (composer, formatting) { if (formatting) { formatting.addButtonDispatch('poll', function (/* textarea */) { onPollButtonClicked(composer); }); + + updateComposerBadgeCount(postContainer, postData.polls ? postData.polls.length : 0); } }); } @@ -28,6 +30,7 @@ const canCreate = await socket.emit('plugins.poll.canCreate', { cid: postData.cid, + pid: postData.pid, tid: postData.tid, }); if (!canCreate) { @@ -64,7 +67,8 @@ postData.polls.push(newPoll); } openManageModal({ ...payload, modal }); - updateComposerBadgeCount(uuid, postData.polls.length); + const postContainer = $(`[component="composer"][data-uuid="${uuid}"]`); + updateComposerBadgeCount(postContainer, postData.polls.length); }); return false; }, @@ -93,7 +97,8 @@ if (poll) { postData.polls = postData.polls.filter(poll => String(poll.pollId) !== clickedPollId); openManageModal(payload); - updateComposerBadgeCount(uuid, postData.polls.length); + const postContainer = $(`[component="composer"][data-uuid="${uuid}"]`); + updateComposerBadgeCount(postContainer, postData.polls.length); } }); }); @@ -241,11 +246,9 @@ }); } - function updateComposerBadgeCount(uuid, count) { - const postContainer = $(`[component="composer"][data-uuid="${uuid}"]`); + function updateComposerBadgeCount(postContainer, count) { require(['composer'], (composer) => { composer.updateFormattingBtnBadgeCount( - uuid, postContainer, 'poll', count From 70bdbf7d17f241864f945169f1598499bc7a1c48 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Sun, 8 Mar 2026 01:09:06 -0500 Subject: [PATCH 03/14] acp settings fixes --- public/js/poll/creator.js | 1 + templates/admin/plugins/poll.tpl | 16 ++++++++-------- templates/poll/creator.tpl | 2 +- upgrades/poll-4.0-upgrade.js | 2 +- 4 files changed, 11 insertions(+), 10 deletions(-) diff --git a/public/js/poll/creator.js b/public/js/poll/creator.js index 81b3e9b..541f2f8 100755 --- a/public/js/poll/creator.js +++ b/public/js/poll/creator.js @@ -62,6 +62,7 @@ // used to identify the poll in the manage modal after creation // so we can edit/delete pollId: Date.now(), + maximumVotesPerUser: config.poll.maximumVotesPerUser, }, config.poll).then((newPoll) => { if (newPoll) { postData.polls.push(newPoll); diff --git a/templates/admin/plugins/poll.tpl b/templates/admin/plugins/poll.tpl index 496e4c9..29abb5d 100755 --- a/templates/admin/plugins/poll.tpl +++ b/templates/admin/plugins/poll.tpl @@ -19,16 +19,16 @@
[[poll:toggles]]
- - + +
[[poll:limits]]
- - + +
@@ -38,12 +38,12 @@
[[poll:defaults]]
- - + +
- - + +

[[poll:info_choices]]

diff --git a/templates/poll/creator.tpl b/templates/poll/creator.tpl index 394c000..778788c 100755 --- a/templates/poll/creator.tpl +++ b/templates/poll/creator.tpl @@ -22,7 +22,7 @@
- +

[[poll:info_choices]]

diff --git a/upgrades/poll-4.0-upgrade.js b/upgrades/poll-4.0-upgrade.js index 6e94abd..4741de4 100644 --- a/upgrades/poll-4.0-upgrade.js +++ b/upgrades/poll-4.0-upgrade.js @@ -51,7 +51,7 @@ async function convertSettings() { newSettings.defaultTitle = parsed.defaults.title; } if (parsed?.defaults?.maxvotes) { - newSettings.maximumVotesPerUser = parsed.defaults.title; + newSettings.maximumVotesPerUser = parsed.defaults.maxvotes; } await db.setObject('settings:poll', newSettings); } catch (err) { From 0cbc1cd5cd42188f632f82ccf5a989b1d1c28209 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Sun, 8 Mar 2026 12:15:20 -0400 Subject: [PATCH 04/14] feat: poll editing --- lib/hooks.js | 46 +++++++---------- lib/poll.js | 97 +++++++++++++++++++++++++++++++++--- lib/vote.js | 22 ++++---- polls-todo.md | 1 + public/js/poll/main.js | 12 ++--- upgrades/poll-4.0-upgrade.js | 2 +- 6 files changed, 125 insertions(+), 55 deletions(-) diff --git a/lib/hooks.js b/lib/hooks.js index 64e9b53..ac2730f 100755 --- a/lib/hooks.js +++ b/lib/hooks.js @@ -42,7 +42,10 @@ Hooks.filter.postCreate = async function (hookData) { // post is going to be saved to db, data is what is submitted by user const { post, data } = hookData; if (Array.isArray(data?.polls) && data.polls.length) { - await Poll.add(post, data.polls); + const savedPolls = await Poll.add(post, data.polls); + if (savedPolls.length) { + post.pollIds = JSON.stringify(savedPolls.map(p => p.pollId)); + } } return hookData; }; @@ -61,37 +64,22 @@ Hooks.filter.postGetFields = async function (hookData) { return hookData; }; -Hooks.filter.postEdit = async function (obj) { - const { tid, pollId } = await posts.getPostFields(obj.data.pid, ['tid', 'pollId']); - if (pollId) { - return obj; - } +Hooks.filter.postEdit = async function (hookData) { + const currentPollIds = await Poll.getPollIdsByPid(hookData.data.pid); + const toAdd = hookData.data.polls.filter(p => !currentPollIds.includes(String(p.pollId))); - const result = await topics.getTopicFields(tid, ['mainPid', 'cid']); - if (parseInt(result.mainPid, 10) !== parseInt(obj.data.pid, 10)) { - return obj; + if (toAdd.length) { + const cid = await posts.getCidByPid(hookData.data.pid, 'cid'); + await canCreate(cid, hookData.data.uid); } - await canCreate(result.cid, obj.post.editor); - - const postData = await savePoll({ - ...obj.post, - uid: obj.data.uid, - pid: obj.data.pid, - tid: tid, - }); - delete postData.uid; - delete postData.pid; - delete postData.tid; - obj.post = postData; - - if (!postData.pollId) { - return obj; - } - - // NodeBB only updates the edited, editor and content fields, so we add the pollId field manually. - await posts.setPostField(obj.data.pid, 'pollId', postData.pollId); - return obj; + await Poll.edit({ + ...hookData.post, + uid: hookData.data.uid, + pid: hookData.data.pid, + }, hookData.data.polls); + hookData.post.pollIds = JSON.stringify(hookData.data.polls.map(p => p.pollId)); + return hookData; }; Hooks.filter.topicPost = async function (data) { diff --git a/lib/poll.js b/lib/poll.js index 8b5ef9c..e6f1570 100755 --- a/lib/poll.js +++ b/lib/poll.js @@ -37,16 +37,99 @@ Poll.add = async function (postData, polls) { savePolls.push(poll); } - if (savePolls.length) { - postData.pollIds = JSON.stringify(savePolls.map(p => p.pollId)); - } - await Promise.all([ db.setObjectBulk(savePolls.map(p => [`poll:${p.pollId}`, p])), db.sortedSetAdd('polls:createtime', savePolls.map(p => p.timestamp), savePolls.map(p => p.pollId)), ]); + return savePolls; +}; + +async function deletePolls(pollIds) { + if (!pollIds.length) return; + const polls = await db.getObjects(pollIds.map(pollId => `poll:${pollId}`)); + const bulkDelete = []; + polls.forEach((poll) => { + if (poll && poll.options) { + const options = tryParseOptions(poll.options); + options.forEach((option) => { + bulkDelete.push(`poll:${poll.pollId}:options:${option.id}:votes`); + }); + } + }); + + bulkDelete.push(...pollIds.map(pollId => `poll:${pollId}`)); + bulkDelete.push(...pollIds.map(pollId => `poll:${pollId}:voters`)); + bulkDelete.push(...pollIds.map(pollId => `poll:${pollId}:anon:voters`)); + await Promise.all([ + db.deleteAll(bulkDelete), + db.sortedSetRemove('polls:createtime', pollIds), + ]); +} + +Poll.edit = async function (postData, newPolls) { + // figure out which polls are new, which are edited and which are deleted + const newPollIds = newPolls.map(p => String(p.pollId)); + const currentPollIds = await Poll.getPollIdsByPid(postData.pid); + const toAdd = newPolls.filter(p => !currentPollIds.includes(String(p.pollId))); + const toUpdate = newPolls.filter(p => currentPollIds.includes(String(p.pollId))); + const toDelete = currentPollIds.filter(pollId => !newPollIds.includes(String(pollId))); + const currentPolls = await db.getObjects(currentPollIds.map(pollId => `poll:${pollId}`)); + + await deletePolls(toDelete); + + await Poll.add(postData, toAdd); + + const bulkUpdateOptions = []; + const bulkDeleteOptions = []; + toUpdate.forEach((poll) => { + if (poll) { + const currentPoll = currentPolls.find(p => String(p.pollId) === String(poll.pollId)); + if (currentPoll) { + const currentOptions = tryParseOptions(currentPoll.options); + const newOptions = poll.options || []; + currentOptions.forEach((option) => { + if (!newOptions.find(o => String(o.id) === String(option.id))) { + bulkDeleteOptions.push(`poll:${poll.pollId}:options:${option.id}:votes`); + } + }); + } + + bulkUpdateOptions.push([`poll:${poll.pollId}`, { + title: poll.title, + end: poll.end || 0, + maximumVotesPerUser: poll.maximumVotesPerUser || 1, + disallowVoteUpdate: poll.disallowVoteUpdate || 0, + allowAnonVoting: poll.allowAnonVoting || 0, + options: JSON.stringify(poll.options || []), + }]); + } + }); + await db.setObjectBulk(bulkUpdateOptions); + await db.deleteAll(bulkDeleteOptions); + + await Promise.all(toUpdate.map(poll => updateVotersFromOptions(poll))); + + if (newPollIds.length) { + await db.setObjectField(`post:${postData.pid}`, 'pollIds', JSON.stringify(newPollIds)); + } else { + await db.deleteObjectField(`post:${postData.pid}`, 'pollIds'); + } }; +async function updateVotersFromOptions(poll) { + // this function updates `poll:${pollId}:voters` sorted set based on the votes on options, + // this is needed when options are removed from a poll via edit + const options = poll.options || []; + + const currentVoters = await db.getSortedSetsMembers( + options.map(option => `poll:${poll.pollId}:options:${option.id}:votes`) + ); + const currentVoterSet = [...new Set(currentVoters.flat())]; + const timestamps = await db.sortedSetScores(`poll:${poll.pollId}:voters`, currentVoterSet); + await db.delete(`poll:${poll.pollId}:voters`); + await db.sortedSetAdd(`poll:${poll.pollId}:voters`, timestamps, currentVoterSet); +} + Poll.get = async function (pollId, uid, withVotes = false) { const [info, hasVoted, vote] = await Promise.all([ Poll.getInfo(pollId, withVotes), @@ -74,7 +157,8 @@ Poll.getPollIdsByPid = async function (pid) { return []; } try { - return JSON.parse(pollIdsJson || '[]'); + const pollIds = JSON.parse(pollIdsJson || '[]'); + return Array.isArray(pollIds) ? pollIds.map(p => String(p)) : []; } catch (err) { return []; } @@ -121,7 +205,8 @@ async function loadOptions(pollId, optionsJson, withVotes = false) { function tryParseOptions(optionsJson) { if (!optionsJson) return []; try { - return JSON.parse(optionsJson || '[]'); + const optionsArray = JSON.parse(optionsJson || '[]'); + return Array.isArray(optionsArray) ? optionsArray : []; } catch (err) { return []; } diff --git a/lib/vote.js b/lib/vote.js index 7d24283..61acaec 100755 --- a/lib/vote.js +++ b/lib/vote.js @@ -1,23 +1,23 @@ 'use strict'; -const NodeBB = require('./nodebb'); +const db = require.main.require('./src/database'); const Poll = require('./poll'); const Vote = exports; Vote.add = async function (voteData) { - const { pollId, options, uid } = voteData; + const { pollId, options, uid, voteAnon } = voteData; const timestamp = Date.now(); const promises = [ - NodeBB.db.sortedSetAdd(`poll:${pollId}:voters`, timestamp, uid), - NodeBB.db.sortedSetAddBulk( + db.sortedSetAdd(`poll:${pollId}:voters`, timestamp, uid), + db.sortedSetAddBulk( options.map(option => ([`poll:${pollId}:options:${option}:votes`, timestamp, uid])) ), ]; - if (voteData.voteAnon) { + if (voteAnon) { promises.push( - NodeBB.db.sortedSetAdd(`poll:${pollId}:anon:voters`, timestamp, uid) + db.sortedSetAdd(`poll:${pollId}:anon:voters`, timestamp, uid) ); } @@ -36,11 +36,11 @@ Vote.removeUidVote = async function (uid, pollId) { const vote = await Vote.getUidVote(uid, pollId); const options = vote.options || []; await Promise.all([ - NodeBB.db.sortedSetsRemove([ + db.sortedSetsRemove([ `poll:${pollId}:voters`, `poll:${pollId}:anon:voters`, ], uid), - NodeBB.db.sortedSetsRemove( + db.sortedSetsRemove( options.map(option => `poll:${pollId}:options:${option}:votes`), uid, ), @@ -60,7 +60,7 @@ Vote.getUidVote = async function (uid, pollId) { Vote.update = async function (voteData) { const { uid, pollId } = voteData; - const isVotedAnon = await NodeBB.db.isSortedSetMember(`poll:${pollId}:anon:voters`, uid); + const isVotedAnon = await db.isSortedSetMember(`poll:${pollId}:anon:voters`, uid); await Vote.remove(voteData); voteData.voteAnon = isVotedAnon; await Vote.add(voteData); @@ -85,12 +85,12 @@ Vote.canVote = async function (uid, pollId) { }; Vote.hasUidVoted = async function (uid, pollId) { - const score = await NodeBB.db.sortedSetScore(`poll:${pollId}:voters`, uid); + const score = await db.sortedSetScore(`poll:${pollId}:voters`, uid); return !!score; }; Vote.hasUidVotedOnOption = async function (uid, pollId, option) { - const score = await NodeBB.db.sortedSetScore(`poll:${pollId}:options:${option}:votes`, uid); + const score = await db.sortedSetScore(`poll:${pollId}:options:${option}:votes`, uid); return !!score; }; diff --git a/polls-todo.md b/polls-todo.md index 9b484b6..dea6088 100644 --- a/polls-todo.md +++ b/polls-todo.md @@ -10,4 +10,5 @@ ability to order poll options by drag and drop TODO: save/restore drafts + (done) edit polls in posts (done) composer formatting toolbar badge number to show how many polls there are \ No newline at end of file diff --git a/public/js/poll/main.js b/public/js/poll/main.js index 443634f..49f617c 100755 --- a/public/js/poll/main.js +++ b/public/js/poll/main.js @@ -19,11 +19,9 @@ $(document).ready(function () { }); $(window).on('action:topic.loading', function () { - if (ajaxify.data.posts.length > 0) { - ajaxify.data.posts.forEach((post) => { - getPollByPost(post); - }); - } + ajaxify.data.posts.forEach((post) => { + getPollByPost(post); + }); }); $(window).on('action:posts.loaded', function (ev, data) { @@ -33,9 +31,7 @@ $(document).ready(function () { }); $(window).on('action:posts.edited', function (ev, data) { - if (data.post.hasOwnProperty('pollId')) { - getPollByPost(data.post); - } + getPollByPost(data.post); }); $(window).on('action:ajaxify.end', function () { diff --git a/upgrades/poll-4.0-upgrade.js b/upgrades/poll-4.0-upgrade.js index 4741de4..4285d94 100644 --- a/upgrades/poll-4.0-upgrade.js +++ b/upgrades/poll-4.0-upgrade.js @@ -71,7 +71,7 @@ async function moveSettingsToPollHash(pollIds) { end: Object.hasOwn(s, 'end') ? s.end : 0, disallowVoteUpdate: Object.hasOwn(s, 'disallowVoteUpdate') ? s.disallowVoteUpdate : 0, allowAnonVoting: Object.hasOwn(s, 'allowAnonVoting') ? s.allowAnonVoting : 0, - maximumVotesPeruser: Object.hasOwn(s, 'maxvotes') ? s.maxvotes : 0, + maximumVotesPerUser: Object.hasOwn(s, 'maxvotes') ? s.maxvotes : 0, }]); } }); From 074f4336ac48ad9aecf50236b5667b63c4bf6b18 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Sun, 8 Mar 2026 14:34:48 -0400 Subject: [PATCH 05/14] fix drafts, add posts.purge hook --- lib/hooks.js | 9 +++++ lib/poll.js | 21 +++++++++-- plugin.json | 1 + polls-todo.md | 8 +++-- public/js/poll/main.js | 40 ++++++++++++++++----- public/js/poll/view.js | 76 ++++++++++++++++++++------------------- templates/poll/manage.tpl | 2 +- 7 files changed, 104 insertions(+), 53 deletions(-) diff --git a/lib/hooks.js b/lib/hooks.js index ac2730f..620393e 100755 --- a/lib/hooks.js +++ b/lib/hooks.js @@ -111,6 +111,15 @@ Hooks.action.postRestore = async function (data) { } }; +Hooks.action.postsPurge = async function (data) { + const { posts } = data; + const pollIds = await Poll.getPollIdsByPids(posts.map(p => p.pid)); + const toDelete = pollIds.flat(); + if (toDelete.length) { + await Poll.deletePolls(toDelete); + } +}; + async function canCreate(cid, uid) { const can = await privileges.categories.can('poll:create', cid, uid); if (!can) { diff --git a/lib/poll.js b/lib/poll.js index e6f1570..85e1778 100755 --- a/lib/poll.js +++ b/lib/poll.js @@ -44,7 +44,7 @@ Poll.add = async function (postData, polls) { return savePolls; }; -async function deletePolls(pollIds) { +Poll.deletePolls = async function (pollIds) { if (!pollIds.length) return; const polls = await db.getObjects(pollIds.map(pollId => `poll:${pollId}`)); const bulkDelete = []; @@ -64,7 +64,7 @@ async function deletePolls(pollIds) { db.deleteAll(bulkDelete), db.sortedSetRemove('polls:createtime', pollIds), ]); -} +}; Poll.edit = async function (postData, newPolls) { // figure out which polls are new, which are edited and which are deleted @@ -75,7 +75,7 @@ Poll.edit = async function (postData, newPolls) { const toDelete = currentPollIds.filter(pollId => !newPollIds.includes(String(pollId))); const currentPolls = await db.getObjects(currentPollIds.map(pollId => `poll:${pollId}`)); - await deletePolls(toDelete); + await Poll.deletePolls(toDelete); await Poll.add(postData, toAdd); @@ -164,6 +164,21 @@ Poll.getPollIdsByPid = async function (pid) { } }; +Poll.getPollIdsByPids = async function (pids) { + const postData = await NodeBB.db.getObjectsFields(pids.map(pid => `post:${pid}`), ['pollIds']); + return postData.map((post) => { + if (!post || !post.pollIds) { + return []; + } + try { + const pollIds = JSON.parse(post.pollIds || '[]'); + return Array.isArray(pollIds) ? pollIds.map(p => String(p)) : []; + } catch (err) { + return []; + } + }); +}; + Poll.getInfo = async function (pollId, withVotes = false) { const [poll, voteCount] = await Promise.all([ db.getObject(`poll:${pollId}`), diff --git a/plugin.json b/plugin.json index 51b7cc9..6568bab 100755 --- a/plugin.json +++ b/plugin.json @@ -20,6 +20,7 @@ { "hook": "action:post.delete", "method": "hooks.action.postDelete" }, { "hook": "action:post.restore", "method": "hooks.action.postRestore" }, + { "hook": "action:posts.purge", "method": "hooks.action.postsPurge" }, { "hook": "filter:widgets.getWidgets", "method": "defineWidgets" }, { "hook": "filter:widget.render:poll", "method": "renderPollWidget" } diff --git a/polls-todo.md b/polls-todo.md index dea6088..03110b9 100644 --- a/polls-todo.md +++ b/polls-todo.md @@ -2,13 +2,15 @@ (done) ability to create more than 1 poll per post -ability to edit polls +(done) ability to edit polls -ability to order polls in a post by drag and drop +(done) ability to order polls in a post by drag and drop ability to order poll options by drag and drop TODO: - save/restore drafts + widget poll + (done) posts purge + (done) save/restore drafts (done) edit polls in posts (done) composer formatting toolbar badge number to show how many polls there are \ No newline at end of file diff --git a/public/js/poll/main.js b/public/js/poll/main.js index 49f617c..0d807dd 100755 --- a/public/js/poll/main.js +++ b/public/js/poll/main.js @@ -16,6 +16,31 @@ $(document).ready(function () { hooks.on('filter:composer.submit', function (hookData) { hookData.composerData.polls = hookData.postData.polls || []; }); + + hooks.on('filter:composer.drafts.save', function (hookData) { + const { draft, postData } = hookData; + if (Object.hasOwn(postData, 'polls')) { + draft.polls = postData.polls || []; + } + return hookData; + }); + + hooks.on('filter:composer.drafts.open', function (hookData) { + const { draft } = hookData; + if (Object.hasOwn(draft, 'polls')) { + hookData.composerData.polls = draft.polls || []; + } + return hookData; + }); + + hooks.on('filter:composer.topic.push', function (hookData) { + hookData.pushData.polls = hookData.data.polls || []; + return hookData; + }); + hooks.on('filter:composer.reply.push', function (hookData) { + hookData.pushData.polls = hookData.data.polls || []; + return hookData; + }); }); $(window).on('action:topic.loading', function () { @@ -60,19 +85,16 @@ $(document).ready(function () { function getPolls(pollIds, container) { const validPollIds = pollIds.map(id => parseInt(id, 10)).filter(id => !isNaN(id)); - if (!socket.connected) { - socket.connect(); - } - - socket.emit('plugins.poll.get', { pollIds: validPollIds.reverse() }, function (err, pollDataArr) { + socket.emit('plugins.poll.get', { pollIds: validPollIds }, async function (err, pollDataArr) { if (err) { return Poll.alertError(err.message); } - - pollDataArr.forEach((pollData) => { + pollDataArr.reverse(); + for (const pollData of pollDataArr) { pollData.container = container; - Poll.view.load(pollData); - }); + // eslint-disable-next-line no-await-in-loop + await Poll.view.load(pollData); + } }); } }); diff --git a/public/js/poll/view.js b/public/js/poll/view.js index 73f93cc..e050aa9 100644 --- a/public/js/poll/view.js +++ b/public/js/poll/view.js @@ -172,43 +172,42 @@ this.pollData = pollData; }; - View.prototype.load = function () { + View.prototype.load = async function () { const self = this; if (!self.pollData.container.length) { return; } - app.parseAndTranslate('poll/view', { poll: self.pollData }, function (panel) { - self.pollData.container.prepend(panel); - self.dom = { - panel: panel, - votingForm: panel.find('.poll-voting-form'), - messages: panel.find('.poll-view-messages'), - votingPanel: panel.find('.poll-view-voting'), - resultsPanel: panel.find('.poll-view-results'), - voteButton: panel.find('.poll-button-vote'), - voteAnonButton: panel.find('.poll-button-vote-anon'), - updateVoteButton: panel.find('.poll-button-update-vote'), - removeVoteButton: panel.find('.poll-button-remove-vote'), - votingPanelButton: panel.find('.poll-button-voting'), - resultsPanelButton: panel.find('.poll-button-results'), - editButton: panel.find('.poll-button-edit'), - }; - - self.hideMessage(); - - self.pollEndedOrDeleted(); - self.hasVotedAndVotingUpdateDisallowed(); - - if (!app.user.uid || self.pollData.hasVoted) { - self.showResultsPanel(); - } else { - self.showVotingPanel(); - } + const panel = await app.parseAndTranslate('poll/view', { poll: self.pollData }); + self.pollData.container.prepend(panel); + self.dom = { + panel: panel, + votingForm: panel.find('.poll-voting-form'), + messages: panel.find('.poll-view-messages'), + votingPanel: panel.find('.poll-view-voting'), + resultsPanel: panel.find('.poll-view-results'), + voteButton: panel.find('.poll-button-vote'), + voteAnonButton: panel.find('.poll-button-vote-anon'), + updateVoteButton: panel.find('.poll-button-update-vote'), + removeVoteButton: panel.find('.poll-button-remove-vote'), + votingPanelButton: panel.find('.poll-button-voting'), + resultsPanelButton: panel.find('.poll-button-results'), + editButton: panel.find('.poll-button-edit'), + }; + + self.hideMessage(); + + self.pollEndedOrDeleted(); + self.hasVotedAndVotingUpdateDisallowed(); + + if (!app.user.uid || self.pollData.hasVoted) { + self.showResultsPanel(); + } else { + self.showVotingPanel(); + } - Actions.forEach(function (action) { - action.register(self); - }); + Actions.forEach(function (action) { + action.register(self); }); }; @@ -398,16 +397,19 @@ }; Poll.view = { - polls: {}, - load: function (pollData) { + polls: Object.create(null), + load: async function (pollData) { + const pollId = pollData.info.pollId; + if (Object.hasOwn(this.polls, pollId)) { + pollData.container.find(`[data-poll-id="${pollId}"]`).remove(); + } const view = new View(pollData); - this.polls[pollData.info.pollId] = view; - - view.load(); + this.polls[pollId] = view; + await view.load(); }, update: function (pollData, uid) { const pollId = pollData.info.pollId; - if (this.polls.hasOwnProperty(pollId)) { + if (Object.hasOwn(this.polls, pollId)) { this.polls[pollId].update(pollData, uid); } }, diff --git a/templates/poll/manage.tpl b/templates/poll/manage.tpl index 63de4de..bd5b253 100644 --- a/templates/poll/manage.tpl +++ b/templates/poll/manage.tpl @@ -3,7 +3,7 @@
[[poll:no-polls]]
{{{ end }}} {{{ each post.polls }}} -
+
{./title}
From 1eecf32405f68f3564c9bb59c5099d8bea493ae0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Sun, 8 Mar 2026 17:35:04 -0400 Subject: [PATCH 06/14] fix: editing and adding a new poll allow sorting options dont crash if poll data is null update readme --- README.md | 21 ++------------------- lib/hooks.js | 5 +++-- lib/poll.js | 20 +++++++++++++++++--- lib/sockets.js | 3 +++ public/js/poll/creator.js | 21 ++++++++++++++++++--- templates/poll/creator.tpl | 2 ++ templates/poll/option-input.tpl | 3 ++- 7 files changed, 47 insertions(+), 28 deletions(-) diff --git a/README.md b/README.md index ecd6dff..36eb027 100644 --- a/README.md +++ b/README.md @@ -1,27 +1,10 @@ # NodeBB Poll plugin -This NodeBB plugin will allow you to add polls to the first post of a topic with the following markup: +This NodeBB plugin will allow you to add polls to posts. - [poll ] - - Poll option - - Another option - [/poll] - -Currently supported settings: - - maxvotes="1" //Max number of votes per user. If larger than 1, a multiple choice poll will be created - disallowVoteUpdate="0" //if set, users won't be able to update/remove their vote - allowAnonVoting="0" // if set to 1, users will be able to vote anonymously - title="Poll title" //Poll title - -There's also a helpful modal available that will allow you to easily create a poll: +There's helpful modal available that will allow you to easily create a poll: ![](https://i.imgur.com/2fPnWLb.png) -## Todo - -- Add the ability to edit a poll -- A lot more... - If you're willing to help, please make any improvements you want and submit a PR. ## Installation diff --git a/lib/hooks.js b/lib/hooks.js index 620393e..11eff59 100755 --- a/lib/hooks.js +++ b/lib/hooks.js @@ -30,6 +30,7 @@ Hooks.filter.registerFormatting = function (payload) { }; Hooks.filter.composerPush = async function (hookData) { + // used for editing, add the polls so they are avable in composer if (hookData.pid) { const pollIds = await Poll.getPollIdsByPid(hookData.pid); const polls = await Promise.all(pollIds.map(pollId => Poll.get(pollId, 0))); @@ -44,7 +45,7 @@ Hooks.filter.postCreate = async function (hookData) { if (Array.isArray(data?.polls) && data.polls.length) { const savedPolls = await Poll.add(post, data.polls); if (savedPolls.length) { - post.pollIds = JSON.stringify(savedPolls.map(p => p.pollId)); + post.pollIds = JSON.stringify(savedPolls.map(p => String(p.pollId))); } } return hookData; @@ -78,7 +79,7 @@ Hooks.filter.postEdit = async function (hookData) { uid: hookData.data.uid, pid: hookData.data.pid, }, hookData.data.polls); - hookData.post.pollIds = JSON.stringify(hookData.data.polls.map(p => p.pollId)); + hookData.post.pollIds = JSON.stringify(hookData.data.polls.map(p => String(p.pollId))); return hookData; }; diff --git a/lib/poll.js b/lib/poll.js index 85e1778..1b7ce48 100755 --- a/lib/poll.js +++ b/lib/poll.js @@ -77,7 +77,12 @@ Poll.edit = async function (postData, newPolls) { await Poll.deletePolls(toDelete); - await Poll.add(postData, toAdd); + const savedPolls = await Poll.add(postData, toAdd); + toAdd.forEach((poll, index) => { + // overwrite with correct pollId generated in add function + // this also updates items in newPolls array with correct pollId so that it can be saved to post object at the end + poll.pollId = savedPolls[index].pollId; + }); const bulkUpdateOptions = []; const bulkDeleteOptions = []; @@ -110,7 +115,11 @@ Poll.edit = async function (postData, newPolls) { await Promise.all(toUpdate.map(poll => updateVotersFromOptions(poll))); if (newPollIds.length) { - await db.setObjectField(`post:${postData.pid}`, 'pollIds', JSON.stringify(newPollIds)); + await db.setObjectField( + `post:${postData.pid}`, + 'pollIds', + JSON.stringify(newPolls.map(p => p.pollId)) + ); } else { await db.deleteObjectField(`post:${postData.pid}`, 'pollIds'); } @@ -136,7 +145,9 @@ Poll.get = async function (pollId, uid, withVotes = false) { uid ? Vote.hasUidVoted(uid, pollId) : false, uid ? Vote.getUidVote(uid, pollId) : null, ]); - + if (!info) { + return null; + } info.options.forEach((option) => { const percentage = ((option.voteCount / info.voteCount) * 100).toFixed(0); option.percentage = isNaN(percentage) ? 0 : percentage; @@ -184,6 +195,9 @@ Poll.getInfo = async function (pollId, withVotes = false) { db.getObject(`poll:${pollId}`), Poll.getVoteCount(pollId), ]); + if (!poll) { + return null; + } poll.voteCount = parseInt(voteCount, 10) || 0; const end = parseInt(poll.end, 10); poll.ended = end > 0 && Date.now() > end; diff --git a/lib/sockets.js b/lib/sockets.js index 0c27649..16b50fd 100755 --- a/lib/sockets.js +++ b/lib/sockets.js @@ -27,6 +27,9 @@ Sockets.get = async function (socket, data) { } return await Promise.all(data.pollIds.map(async (pollId) => { const pollData = await Poll.get(pollId, socket.uid, !!socket.uid); + if (!pollData) { + return null; + } if (parseInt(pollData.allowAnonVoting, 10) === 1) { await anonymizeVoters(socket.uid, pollId, pollData.options); diff --git a/public/js/poll/creator.js b/public/js/poll/creator.js index 541f2f8..4f715a4 100755 --- a/public/js/poll/creator.js +++ b/public/js/poll/creator.js @@ -41,6 +41,7 @@ } async function openManageModal(payload) { + const bootbox = await app.require('bootbox'); const { postData, modal, uuid } = payload; const html = await app.parseAndTranslate('poll/manage', { post: postData }); @@ -86,7 +87,8 @@ } } - function handleDelete(payload) { + async function handleDelete(payload) { + const bootbox = await app.require('bootbox'); const { uuid, modal, postData } = payload; modal.on('click', 'button[data-action="remove"]', function () { const clickedPollId = $(this).attr('data-poll-id'); @@ -111,7 +113,7 @@ const clickedPollId = $(this).attr('data-poll-id'); const poll = postData?.polls.find(poll => String(poll.pollId) === clickedPollId); if (poll) { - const editedPoll = await Creator.show(poll, payload.config); + const editedPoll = await Creator.show(poll, config.poll); if (editedPoll) { postData.polls = postData.polls.map( p => String(p.pollId) === String(editedPoll.pollId) ? editedPoll : p @@ -183,6 +185,8 @@ }, }); + handleOptionSort({ modal }); + // Add option adder modal.find('#pollAddOption') .off('click') @@ -198,7 +202,8 @@ id: id, title: '', }); - newOptionInput.insertBefore(el); + const container = modal.find('#poll-options-container'); + container.append(newOptionInput); modal.find(`[data-option-id="${id}"] input`).focus(); }); }); @@ -206,6 +211,16 @@ }); }; + function handleOptionSort({ modal }) { + const selectorEl = modal.find('#poll-options-container'); + selectorEl.sortable({ + handle: '[component="sort/handle"]', + axis: 'y', + zIndex: 9999, + items: '[component="post/poll/option/item"]', + }); + } + async function error(message) { const errorBox = $('#pollErrorBox'); const translator = await app.require('translator'); diff --git a/templates/poll/creator.tpl b/templates/poll/creator.tpl index 778788c..f89afe3 100755 --- a/templates/poll/creator.tpl +++ b/templates/poll/creator.tpl @@ -10,11 +10,13 @@
+
{{{ if poll.options.length }}} {{{ each poll.options }}} {{{ end }}} {{{ end }}} +
diff --git a/templates/poll/option-input.tpl b/templates/poll/option-input.tpl index 833d33b..834394c 100644 --- a/templates/poll/option-input.tpl +++ b/templates/poll/option-input.tpl @@ -1,4 +1,5 @@ -
+
+ From 031ef3fe819ba4b8fafbe707c683f8127a102dd8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Sun, 8 Mar 2026 18:13:58 -0400 Subject: [PATCH 07/14] various fixes show different title in edit/create fix widget mode link to post on widget poll title remove old edit button --- languages/en_GB/poll.json | 2 ++ lib/hooks.js | 2 +- polls-todo.md | 16 ---------------- public/js/poll/creator.js | 12 ++++++++---- public/js/poll/main.js | 6 +++++- public/js/poll/view.js | 23 ++++------------------- templates/admin/partials/widgets/poll.tpl | 4 ++-- templates/poll/creator.tpl | 6 ------ templates/poll/manage.tpl | 4 ++-- templates/poll/option-input.tpl | 4 ++-- templates/poll/view.tpl | 19 ++++++++++++------- 11 files changed, 38 insertions(+), 60 deletions(-) delete mode 100644 polls-todo.md diff --git a/languages/en_GB/poll.json b/languages/en_GB/poll.json index 32cb87c..cf0ab90 100755 --- a/languages/en_GB/poll.json +++ b/languages/en_GB/poll.json @@ -1,5 +1,6 @@ { "poll": "Poll", + "poll-id-x": "Poll ID: %1", "toggles": "Toggles", "allow_guests": "Allow guests to view poll results", "limits": "Limits", @@ -18,6 +19,7 @@ "no-polls": "No polls found", "confirm-remove": "Are you sure you want to remove this poll?", "creator_title": "Create a poll", + "edit-a-poll-title": "Edit a poll", "poll_title": "Poll Title", "poll_title_placeholder": "Enter poll title", "options_title": "Options", diff --git a/lib/hooks.js b/lib/hooks.js index 11eff59..dc81f27 100755 --- a/lib/hooks.js +++ b/lib/hooks.js @@ -23,7 +23,7 @@ Hooks.filter.registerFormatting = function (payload) { payload.options.push({ name: 'poll', className: `fa ${Config.plugin.icon}`, - title: '[[poll:creator_title]]', + title: '[[poll:manage-polls]]', badge: true, }); return payload; diff --git a/polls-todo.md b/polls-todo.md deleted file mode 100644 index 03110b9..0000000 --- a/polls-todo.md +++ /dev/null @@ -1,16 +0,0 @@ -(done) ability to create polls in every post not just main post - -(done) ability to create more than 1 poll per post - -(done) ability to edit polls - -(done) ability to order polls in a post by drag and drop - -ability to order poll options by drag and drop - -TODO: - widget poll - (done) posts purge - (done) save/restore drafts - (done) edit polls in posts - (done) composer formatting toolbar badge number to show how many polls there are \ No newline at end of file diff --git a/public/js/poll/creator.js b/public/js/poll/creator.js index 4f715a4..52d7b9f 100755 --- a/public/js/poll/creator.js +++ b/public/js/poll/creator.js @@ -58,7 +58,7 @@ label: ' [[poll:add-poll]]', className: 'btn-success', callback: () => { - Creator.show({ + Creator.show('[[poll:creator_title]]', { // temp id, will be replaced in the backend before saving // used to identify the poll in the manage modal after creation // so we can edit/delete @@ -113,7 +113,11 @@ const clickedPollId = $(this).attr('data-poll-id'); const poll = postData?.polls.find(poll => String(poll.pollId) === clickedPollId); if (poll) { - const editedPoll = await Creator.show(poll, config.poll); + const editedPoll = await Creator.show( + '[[poll:edit-a-poll-title]]', + poll, + config.poll, + ); if (editedPoll) { postData.polls = postData.polls.map( p => String(p.pollId) === String(editedPoll.pollId) ? editedPoll : p @@ -150,12 +154,12 @@ } }; - Creator.show = function (poll, config) { + Creator.show = function (modalTitle, poll, config) { return new Promise((resolve) => { require(['bootbox'], function (bootbox) { app.parseAndTranslate('poll/creator', { poll, config }, function (html) { const modal = bootbox.dialog({ - title: '[[poll:creator_title]]', + title: modalTitle, message: html, className: 'poll-creator', buttons: { diff --git a/public/js/poll/main.js b/public/js/poll/main.js index 0d807dd..8d5aca5 100755 --- a/public/js/poll/main.js +++ b/public/js/poll/main.js @@ -61,7 +61,8 @@ $(document).ready(function () { $(window).on('action:ajaxify.end', function () { $('[data-widget-poll-id]').each(function () { - getPolls([$(this).attr('data-widget-poll-id')], $(this)); + const $this = $(this); + getPolls([$this.attr('data-widget-poll-id')], $this); }); }); @@ -92,6 +93,9 @@ $(document).ready(function () { pollDataArr.reverse(); for (const pollData of pollDataArr) { pollData.container = container; + if (container.attr('data-widget-poll-id')) { + pollData.isWidget = true; + } // eslint-disable-next-line no-await-in-loop await Poll.view.load(pollData); } diff --git a/public/js/poll/view.js b/public/js/poll/view.js index e050aa9..9b8720d 100644 --- a/public/js/poll/view.js +++ b/public/js/poll/view.js @@ -149,23 +149,6 @@ }); }, }, - { - // Editing - register: function (view) { - const self = this; - view.dom.editButton.off('click').on('click', function () { - self.handle(view); - }); - }, - handle: function (view) { - socket.emit('plugins.poll.getConfig', null, function (err, config) { - if (err) { - console.error(err); - } - Poll.creator.show(view.pollData, config, function () {}); - }); - }, - }, ]; const View = function (pollData) { @@ -178,7 +161,10 @@ return; } - const panel = await app.parseAndTranslate('poll/view', { poll: self.pollData }); + const panel = await app.parseAndTranslate('poll/view', { + poll: self.pollData, + isAdmin: app.user.isAdmin, + }); self.pollData.container.prepend(panel); self.dom = { panel: panel, @@ -192,7 +178,6 @@ removeVoteButton: panel.find('.poll-button-remove-vote'), votingPanelButton: panel.find('.poll-button-voting'), resultsPanelButton: panel.find('.poll-button-results'), - editButton: panel.find('.poll-button-edit'), }; self.hideMessage(); diff --git a/templates/admin/partials/widgets/poll.tpl b/templates/admin/partials/widgets/poll.tpl index 6465698..cd7d6bd 100644 --- a/templates/admin/partials/widgets/poll.tpl +++ b/templates/admin/partials/widgets/poll.tpl @@ -1,5 +1,5 @@
- - + +
diff --git a/templates/poll/creator.tpl b/templates/poll/creator.tpl index f89afe3..730c199 100755 --- a/templates/poll/creator.tpl +++ b/templates/poll/creator.tpl @@ -45,10 +45,4 @@

[[poll:auto_end_help]]

- - {{{ if isRedactor }}} - - {{{ end }}} diff --git a/templates/poll/manage.tpl b/templates/poll/manage.tpl index bd5b253..787131b 100644 --- a/templates/poll/manage.tpl +++ b/templates/poll/manage.tpl @@ -3,9 +3,9 @@
[[poll:no-polls]]
{{{ end }}} {{{ each post.polls }}} -
+
- +
{./title}
diff --git a/templates/poll/option-input.tpl b/templates/poll/option-input.tpl index 834394c..5a172f1 100644 --- a/templates/poll/option-input.tpl +++ b/templates/poll/option-input.tpl @@ -1,5 +1,5 @@ -
- +
+ diff --git a/templates/poll/view.tpl b/templates/poll/view.tpl index b93f4ab..4904c57 100644 --- a/templates/poll/view.tpl +++ b/templates/poll/view.tpl @@ -1,11 +1,16 @@
-
-
-
{poll.info.title}
- - +
+
+
+ {{{ if poll.isWidget }}} + {poll.info.title} + {{{ else }}} + {poll.info.title} + {{{ end }}} +
+ {{{ if isAdmin }}} + [[poll:poll-id-x, {poll.info.pollId}]] + {{{ end }}}
From 18b3abdd3156e30debd071958f3312fa24caa8ad Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Sun, 8 Mar 2026 18:20:03 -0400 Subject: [PATCH 08/14] =?UTF-8?q?abililty=20to=20remove=20options,=20use?= =?UTF-8?q?=20at=20your=20own=20risk=20=F0=9F=A4=A3?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- public/js/poll/creator.js | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/public/js/poll/creator.js b/public/js/poll/creator.js index 52d7b9f..c346b06 100755 --- a/public/js/poll/creator.js +++ b/public/js/poll/creator.js @@ -210,6 +210,10 @@ container.append(newOptionInput); modal.find(`[data-option-id="${id}"] input`).focus(); }); + + modal.find('#poll-options-container').on('click', '[data-action="remove-option"]', function () { + $(this).parents('[component="post/poll/option/item"]').remove(); + }); }); }); }); From 4c53edbe56f801e8fbf9f17ac9796c3e1df039ff Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Sun, 8 Mar 2026 18:23:17 -0400 Subject: [PATCH 09/14] use db directly --- lib/poll.js | 13 ++++++------- lib/sockets.js | 2 +- 2 files changed, 7 insertions(+), 8 deletions(-) diff --git a/lib/poll.js b/lib/poll.js index 1b7ce48..2d8ad26 100755 --- a/lib/poll.js +++ b/lib/poll.js @@ -1,6 +1,5 @@ 'use strict'; -const NodeBB = require('./nodebb'); const db = require.main.require('./src/database'); const Vote = require('./vote'); @@ -163,7 +162,7 @@ Poll.get = async function (pollId, uid, withVotes = false) { }; Poll.getPollIdsByPid = async function (pid) { - const pollIdsJson = await NodeBB.db.getObjectField(`post:${pid}`, 'pollIds'); + const pollIdsJson = await db.getObjectField(`post:${pid}`, 'pollIds'); if (!pollIdsJson) { return []; } @@ -176,7 +175,7 @@ Poll.getPollIdsByPid = async function (pid) { }; Poll.getPollIdsByPids = async function (pids) { - const postData = await NodeBB.db.getObjectsFields(pids.map(pid => `post:${pid}`), ['pollIds']); + const postData = await db.getObjectsFields(pids.map(pid => `post:${pid}`), ['pollIds']); return postData.map((post) => { if (!post || !post.pollIds) { return []; @@ -326,18 +325,18 @@ Poll.doesAllowVoteUpdate = async function (pollId) { }; Poll.setField = async function (pollId, field, value) { - await NodeBB.db.setObjectField(`poll:${pollId}`, field, value); + await db.setObjectField(`poll:${pollId}`, field, value); }; Poll.setFields = async function (pollId, data) { - await NodeBB.db.setObject(`poll:${pollId}`, data); + await db.setObject(`poll:${pollId}`, data); }; Poll.getField = async function (pollId, field) { - return await NodeBB.db.getObjectField(`poll:${pollId}`, field); + return await db.getObjectField(`poll:${pollId}`, field); }; Poll.getFields = async function (pollId, fields) { - return await NodeBB.db.getObjectFields(`poll:${pollId}`, fields); + return await db.getObjectFields(`poll:${pollId}`, fields); }; diff --git a/lib/sockets.js b/lib/sockets.js index 16b50fd..70befaa 100755 --- a/lib/sockets.js +++ b/lib/sockets.js @@ -187,7 +187,7 @@ Sockets.canCreate = async function (socket, data) { async function anonymizeVoters(callerUid, pollId, options) { const uids = _.uniq(options.map(opt => opt.votes).flat()); const [isAnon, isPrivileged] = await Promise.all([ - NodeBB.db.isSortedSetMembers(`poll:${pollId}:anon:voters`, uids), + db.isSortedSetMembers(`poll:${pollId}:anon:voters`, uids), NodeBB.User.isPrivileged(callerUid), ]); if (isPrivileged) { From 5a1d15794aa4b6e6eea3e4d87d58751fc936959a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Sun, 8 Mar 2026 18:23:58 -0400 Subject: [PATCH 10/14] remove css --- public/scss/style.scss | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/public/scss/style.scss b/public/scss/style.scss index b65a6f2..8b13789 100644 --- a/public/scss/style.scss +++ b/public/scss/style.scss @@ -1,4 +1 @@ -// Hide poll button in composer on reply -.composer.reply .poll, .composer.reply .ql-poll { - display: none; -} + From 44c3ba7051a557d696759a6e5a19268605a48f42 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Sun, 8 Mar 2026 18:37:37 -0400 Subject: [PATCH 11/14] chore: up compat --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 62b528b..6907120 100755 --- a/package.json +++ b/package.json @@ -26,7 +26,7 @@ "cron": "^1.8.2" }, "nbbpm": { - "compatibility": "^4.0.0" + "compatibility": "^4.9.2" }, "devDependencies": { "eslint": "10.0.3", From 33bd0cb78b05e8c8ed2e6db8db48650dd069ed34 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Sun, 8 Mar 2026 18:44:08 -0400 Subject: [PATCH 12/14] if user doesn't hve privilege to read the post the poll is in throw error --- lib/poll.js | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/lib/poll.js b/lib/poll.js index 2d8ad26..e9bb8a0 100755 --- a/lib/poll.js +++ b/lib/poll.js @@ -1,6 +1,7 @@ 'use strict'; const db = require.main.require('./src/database'); +const privileges = require.main.require('./src/privileges'); const Vote = require('./vote'); @@ -147,6 +148,10 @@ Poll.get = async function (pollId, uid, withVotes = false) { if (!info) { return null; } + const canRead = await privileges.posts.can('topics:read', info.pid, uid); + if (!canRead) { + throw new Error('[[error:no-privileges]]'); + } info.options.forEach((option) => { const percentage = ((option.voteCount / info.voteCount) * 100).toFixed(0); option.percentage = isNaN(percentage) ? 0 : percentage; From 712be813d27067ee1967d291be91f68c8475cd9b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Sun, 8 Mar 2026 18:49:36 -0400 Subject: [PATCH 13/14] add privilege checks if users can read post, dont allow getting poll/options/voting --- lib/sockets.js | 9 ++++++++- lib/vote.js | 21 +++++++++++++++++---- 2 files changed, 25 insertions(+), 5 deletions(-) diff --git a/lib/sockets.js b/lib/sockets.js index 70befaa..8f8b05d 100755 --- a/lib/sockets.js +++ b/lib/sockets.js @@ -4,6 +4,7 @@ const _ = require.main.require('lodash'); const db = require.main.require('./src/database'); const posts = require.main.require('./src/posts'); const topics = require.main.require('./src/topics'); +const privileges = require.main.require('./src/privileges'); const NodeBB = require('./nodebb'); const Config = require('./config'); @@ -150,7 +151,13 @@ Sockets.getOptionDetails = async function (socket, data) { Poll.getInfo(data.pollId), Poll.getOption(data.pollId, data.optionId, true), ]); - + if (!poll) { + throw new Error('[[error:invalid-data]]'); + } + const canRead = await privileges.posts.can('topics:read', poll.pid, socket.uid); + if (!canRead) { + throw new Error('[[error:no-privileges]]'); + } if (!option || !option.votes || !option.votes.length) { return option; } diff --git a/lib/vote.js b/lib/vote.js index 61acaec..5d3a6b6 100755 --- a/lib/vote.js +++ b/lib/vote.js @@ -1,6 +1,8 @@ 'use strict'; const db = require.main.require('./src/database'); +const privileges = require.main.require('./src/privileges'); + const Poll = require('./poll'); const Vote = exports; @@ -67,21 +69,32 @@ Vote.update = async function (voteData) { }; Vote.canUpdateVote = async function (uid, pollId) { - const [ended, deleted, updateDisabled] = await Promise.all([ + const pollInfo = await Poll.getInfo(pollId, false); + if (!pollInfo) { + throw new Error('[[error:invalid-data]]'); + } + const [canRead, ended, deleted, updateDisabled] = await Promise.all([ + privileges.posts.can('topics:read', pollInfo.pid, uid), Poll.hasEnded(pollId), Poll.isDeleted(pollId), Poll.doesDisallowVoteUpdate(pollId), ]); - return !ended && !deleted && !updateDisabled; + return !canRead && !ended && !deleted && !updateDisabled; }; Vote.canVote = async function (uid, pollId) { - const [ended, deleted, hasVoted] = await Promise.all([ + const pollInfo = await Poll.getInfo(pollId, false); + if (!pollInfo) { + throw new Error('[[error:invalid-data]]'); + } + + const [canRead, ended, deleted, hasVoted] = await Promise.all([ + privileges.posts.can('topics:read', pollInfo.pid, uid), Poll.hasEnded(pollId), Poll.isDeleted(pollId), Vote.hasUidVoted(uid, pollId), ]); - return !ended && !deleted && !hasVoted; + return !canRead && !ended && !deleted && !hasVoted; }; Vote.hasUidVoted = async function (uid, pollId) { From 90c8673297e439c9f3b2347b8a710399e6d0cd1a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= Date: Sun, 8 Mar 2026 19:48:57 -0400 Subject: [PATCH 14/14] add acp page to view polls --- languages/en_GB/poll.json | 4 +- library.js | 27 +++++++- templates/admin/plugins/poll.tpl | 111 ++++++++++++++++++------------- upgrades/poll-4.0-upgrade.js | 5 +- 4 files changed, 96 insertions(+), 51 deletions(-) diff --git a/languages/en_GB/poll.json b/languages/en_GB/poll.json index cf0ab90..f6fa895 100755 --- a/languages/en_GB/poll.json +++ b/languages/en_GB/poll.json @@ -55,5 +55,7 @@ "votes": "votes", "x-votes": "%1 votes", "total-votes-x": "Total Votes: %1", - "admin.create-poll": "Create Poll" + "admin.create-poll": "Create Poll", + "created-time": "Created", + "end-time": "End" } \ No newline at end of file diff --git a/library.js b/library.js index 4f30d83..101198f 100755 --- a/library.js +++ b/library.js @@ -4,6 +4,10 @@ const NodeBB = require('./lib/nodebb'); const Config = require('./lib/config'); const Sockets = require('./lib/sockets'); const Hooks = require('./lib/hooks'); +const Poll = require('./lib/poll'); +const db = require.main.require('./src/database'); +const pagination = require.main.require('./src/pagination'); +const utils = require.main.require('./src/utils'); const Plugin = module.exports; @@ -13,9 +17,30 @@ Plugin.load = async function (params) { const routeHelpers = require.main.require('./src/routes/helpers'); const { router } = params; - routeHelpers.setupAdminPageRoute(router, `/admin/plugins/${Config.plugin.id}`, (req, res) => { + routeHelpers.setupAdminPageRoute(router, `/admin/plugins/${Config.plugin.id}`, async (req, res) => { + const page = Math.max(parseInt(req.query.page, 10) || 1, 1); + const itemsPerPage = 20; + const start = Math.max(0, (page - 1) * itemsPerPage); + const stop = start + itemsPerPage - 1; + const [pollIds, count] = await Promise.all([ + db.getSortedSetRevRange('polls:createtime', start, stop), + db.sortedSetCard('polls:createtime'), + ]); + + const pollData = await db.getObjects(pollIds.map(pollId => `poll:${pollId}`)); + const voteCounts = await Promise.all(pollIds.map(Poll.getVoteCount)); + pollData.forEach((poll, index) => { + if (poll) { + poll.timestampISO = utils.toISOString(poll.timestamp); + poll.endISO = utils.toISOString(poll.end); + poll.voteCount = voteCounts[index]; + } + }); + const pageCount = Math.ceil(count / itemsPerPage); res.render(`admin/plugins/${Config.plugin.id}`, { title: 'Poll', + polls: pollData, + pagination: pagination.create(page, pageCount, req.query), }); }); diff --git a/templates/admin/plugins/poll.tpl b/templates/admin/plugins/poll.tpl index 29abb5d..44f2720 100755 --- a/templates/admin/plugins/poll.tpl +++ b/templates/admin/plugins/poll.tpl @@ -1,57 +1,72 @@
-
-

{title}

-
-
- - -
-
+
+

{title}

+
+
+ +
+
-
-
-
-
[[poll:poll]]
-
-
-
[[poll:toggles]]
-
-
- - -
-
-
-
-
[[poll:limits]]
-
- - -
-
-
-
- -
-
[[poll:defaults]]
-
-
- - -
-
- - -

[[poll:info_choices]]

-
-
-
-
+
+
+ + + + + + + + + + + + {{{ each polls }}} + + + + + + + + {{{ end }}} + +
ID[[poll:poll_title]][[poll:created-time]][[poll:end-time]][[poll:votes]]
{./pollId}{./title} + {{{ if ./end }}} + + {{{ else }}} + - + {{{ end }}} + + {./voteCount} +
+
+
+
+
+
+ + +
+ +
+ + +
- +
+ + +
+
+ + +

[[poll:info_choices]]

+
+
+
diff --git a/upgrades/poll-4.0-upgrade.js b/upgrades/poll-4.0-upgrade.js index 4285d94..67efb41 100644 --- a/upgrades/poll-4.0-upgrade.js +++ b/upgrades/poll-4.0-upgrade.js @@ -99,7 +99,10 @@ async function savePollOptionsInPollHash(pollIds) { const optionData = await db.getObjects(keys); bulkSet.push([ `poll:${pollId}`, - { options: JSON.stringify(optionData.map(o => ({ id: o.id, title: o.title }))) }, + { + pollId: pollId, // some old polls had "pollid" field, this fixes that + options: JSON.stringify(optionData.map(o => ({ id: o.id, title: o.title }))) + }, ]); } }));