From 0c14d271c9142cebf222955163e53480fd67f58e Mon Sep 17 00:00:00 2001 From: timothycarambat Date: Fri, 22 Nov 2024 14:15:39 -0800 Subject: [PATCH] Initial commit --- .npmignore | 8 + AnythingLLMStore.js | 137 ++++++ CommunityHubInterface.js | 116 +++++ README.md | 45 ++ bin.js | 112 +++++ package.json | 46 ++ template/agent-skill/handler.js | 40 ++ template/agent-skill/plugin.json | 19 + template/agent-skill/readme.md | 11 + utils/validateAgentSkill.js | 256 ++++++++++ yarn.lock | 792 +++++++++++++++++++++++++++++++ 11 files changed, 1582 insertions(+) create mode 100644 .npmignore create mode 100644 AnythingLLMStore.js create mode 100644 CommunityHubInterface.js create mode 100644 README.md create mode 100755 bin.js create mode 100644 package.json create mode 100644 template/agent-skill/handler.js create mode 100644 template/agent-skill/plugin.json create mode 100644 template/agent-skill/readme.md create mode 100644 utils/validateAgentSkill.js create mode 100644 yarn.lock diff --git a/.npmignore b/.npmignore new file mode 100644 index 0000000..c62f74f --- /dev/null +++ b/.npmignore @@ -0,0 +1,8 @@ +.git +.gitignore +node_modules +npm-debug.log +.DS_Store +*.test.js +.vscode +.idea \ No newline at end of file diff --git a/AnythingLLMStore.js b/AnythingLLMStore.js new file mode 100644 index 0000000..b59caed --- /dev/null +++ b/AnythingLLMStore.js @@ -0,0 +1,137 @@ +const fs = require('fs'); +const path = require('path'); +const os = require('os'); +const input = require('input'); +const CommunityHubInterface = require('./CommunityHubInterface'); +const validateAgentSkill = require('./utils/validateAgentSkill'); + +/** + * AnythingLLMStore is a class that stores the connection key for the AnythingLLM Hub. + */ +class AnythingLLMStore extends CommunityHubInterface { + configPath = path.join(os.homedir(), '.anythingllm-hub-cli-config.json'); + connectionKey = null; + userInfo = null; + supportedUploadTypes = ['agent-skill']; + + /** + * @param {Object} options + * @param {boolean} options.debug + */ + constructor({ debug = false } = {}) { + if (debug) { + console.log(`---`); + console.log(`AnythingLLM CLI initialized with debug: ${debug}`); + console.log(`---`); + } + super({ debug }); + this._loadConfig(); + } + + _loadConfig() { + if (fs.existsSync(this.configPath)) { + const config = JSON.parse(fs.readFileSync(this.configPath, 'utf8')); + + if (config.connectionKey) { + this.connectionKey = config.connectionKey; + this.userInfo = config.userInfo; + console.log(`Loaded connection key: ${this.connectionKey}`); + } else { + console.log('No connection key found in config - you will need to login to push to the AnythingLLM Hub'); + } + } + } + + /** + * Updates the config file with the given updates. + * @param {Object} updates + */ + _saveConfig(updates = {}) { + // If the config file doesn't exist, create it + if (!fs.existsSync(this.configPath)) return fs.writeFileSync(this.configPath, JSON.stringify(updates, null, 2)); + + // Otherwise, update the existing config + const config = JSON.parse(fs.readFileSync(this.configPath, 'utf8')); + Object.assign(config, updates); + fs.writeFileSync(this.configPath, JSON.stringify(config, null, 2)); + } + + _wipeConfig() { + if (!fs.existsSync(this.configPath)) return; + fs.unlinkSync(this.configPath); + } + + /** + * Shows the current config in the console. + */ + showConfig() { + if (!fs.existsSync(this.configPath)) { + console.log('No config file found. Run `anythingllm-hub-cli login` to login and save your connection key.'); + return; + } + console.log(JSON.parse(fs.readFileSync(this.configPath, 'utf8'))); + } + + /** + * Logs in to the AnythingLLM Hub using a connection key. + * and if successful, saves the connection key to the config file. + * @returns {Promise} + */ + async login() { + const connectionKey = await input.text('Enter your AnythingLLM Hub connection key:'); + if (!connectionKey) throw new Error('Connection key is required'); + + const authCheck = await this.authCheck(connectionKey); + if (!authCheck) throw new Error('Invalid connection key - could not authenticate'); + this.connectionKey = connectionKey; + this.userInfo = await this.getUserInfo(); + this._saveConfig({ connectionKey, userInfo: this.userInfo }); + console.log('Successfully logged in!'); + } + + /** + * Logs out of the AnythingLLM Hub. + * @returns {Promise} + */ + async logout() { + this.connectionKey = null; + this.userInfo = null; + this._wipeConfig(); + console.log('Successfully logged out!'); + } + + async upload({ type, path: _folderPath }) { + const folderPath = path.resolve(_folderPath); // Resolve the path to ensure it's absolute + if (!this.connectionKey) throw new Error('No connection key found - you will need to login first to push to the AnythingLLM Hub'); + if (!this.supportedUploadTypes.includes(type)) throw new Error(`Unsupported upload type: ${type}`); + if (!folderPath) throw new Error('--path argument is required'); + if (!fs.existsSync(folderPath)) throw new Error(`--path argument was provided but the path does not exist`); + if (!fs.lstatSync(folderPath).isDirectory()) throw new Error(`--path argument was provided but the path is not a directory`); + + const authCheck = await this.authCheck(this.connectionKey); + if (!authCheck) throw new Error('Invalid connection key - could not authenticate.'); + + if (type === 'agent-skill') return await validateAgentSkill(this, folderPath); + console.log(`Uploading Complete.`); + } + + /** + * Creates a new item in the AnythingLLM Hub. + * @param {Object} args + * @param {string} args.type - The type of item to create + * @param {string} args.output - The path to save the new item to, can be relative to the current working directory + */ + async newItem({ type, output }) { + if (!this.supportedUploadTypes.includes(type)) throw new Error(`Unsupported upload type: ${type}`); + if (!output) throw new Error('--output argument is required'); + if (!fs.existsSync(output)) throw new Error(`--output argument was provided but the path does not exist`); + + const outputFolder = output === '.' ? `my-${type}-${Date.now().toString().slice(-3)}` : output; + const outputFolderPath = path.resolve(outputFolder); + const templatePath = path.join(__dirname, `template/${type}`); + fs.cpSync(templatePath, outputFolderPath, { recursive: true }); + console.log(`New ${type} created at ${outputFolderPath}`); + } +} + +module.exports = AnythingLLMStore; diff --git a/CommunityHubInterface.js b/CommunityHubInterface.js new file mode 100644 index 0000000..ebf3baa --- /dev/null +++ b/CommunityHubInterface.js @@ -0,0 +1,116 @@ +const fetch = require('node-fetch'); +const fs = require('fs'); + +class CommunityHubInterface { + apiBaseDev = "http://127.0.0.1:5001/anythingllm-hub/us-central1/external"; + apiBaseProd = "https://hub.external.anythingllm.com"; + + constructor({ debug = false } = {}) { + this.debug = debug; + this.apiBase = debug ? this.apiBaseDev : this.apiBaseProd; + } + + _debugLog(message) { + if (this.debug) console.log(`[CommunityHubInterface:DEBUG] ${message}`); + } + + _log(message, ...args) { + console.log(`[CommunityHubInterface] ${message}`, ...args); + } + + /** + * Checks if the connection key is valid. + * @param {string} connectionKey + * @returns {Promise} + */ + async authCheck(connectionKey) { + if (!this.debug) this._log(`Logging into hub.anythingllm.com...`); + this._debugLog(`Checking auth for connection key against ${new URL(this.apiBase).origin}`); + return await fetch(`${this.apiBase}/auth`, { + method: 'GET', + headers: { + 'Authorization': `Bearer ${connectionKey}` + } + }) + .then(response => response.ok) + .catch(error => { + console.error('Error checking auth:', error); + return false; + }); + } + + /** + * Gets the user info. + * @returns {Promise<{id: string, username: string, author_url: string}>} + */ + async getUserInfo() { + return fetch(`${this.apiBase}/v1/me`, { + headers: { 'Authorization': `Bearer ${this.connectionKey}` } + }).then(response => response.json()) + .catch(error => { + console.error('Error getting user info:', error); + return null; + }); + } + + /** + * Registers a skill with the AnythingLLM Hub. + * @param {('agent-skill')} entityType - The type of entity to register. + * @returns {Promise<{signedUrl: string, uri: string, entityId: string, error?: string}>} + */ + async prepareUpload(entityType, visibility) { + return fetch(`${this.apiBase}/v1/${entityType}/prepare`, { + method: 'POST', + headers: { 'Authorization': `Bearer ${this.connectionKey}` }, + body: JSON.stringify({ visibility }) + }) + .then(response => { + if (!response.ok) throw new Error(`${response.status} - ${response.statusText}`); + return response.json(); + }) + .catch(error => { + console.error('Error preparing upload:', error); + return { error: error.message }; + }); + } + + /** + * Uploads a file to the AnythingLLM Hub. + * @param {Object} uploadResponse - The response from the prepareUpload method. + * @param {Object} uploadResponse.signedUrl - The signed URL to upload the file to. + * @param {string} filePath - The path to the file to upload. + * @returns {Promise} + */ + async uploadFile({ signedUrl }, zipBundlePath) { + const response = await fetch(signedUrl, { + method: 'PUT', + body: fs.readFileSync(zipBundlePath), + headers: { + 'Content-Type': 'application/zip', + } + }); + return response.ok; + } + + /** + * Finalizes an upload to the AnythingLLM Hub. + * @param {('agent-skill')} entityType - The type of entity to finalize. + * @param {string} entityId - The ID of the entity to finalize. + * @param {{name: string, description: string, files: {path: string, name: string, size: number, content: string}[]}[]} manifestData - The manifest data to finalize the upload with. + * @returns {Promise} + */ + async finalizeUpload(entityType, entityId, manifestData) { + return fetch(`${this.apiBase}/v1/${entityType}/finalize/${entityId}`, { + method: 'POST', + headers: { 'Authorization': `Bearer ${this.connectionKey}` }, + body: JSON.stringify(manifestData) + }) + .then(response => response.ok) + .catch(error => { + console.error('Error finalizing upload:', error); + return false; + }); + } +} + +module.exports = CommunityHubInterface; diff --git a/README.md b/README.md new file mode 100644 index 0000000..89b8d8a --- /dev/null +++ b/README.md @@ -0,0 +1,45 @@ +# AnythingLLM Hub CLI + +## Overview + +[AnythingLLM](https://github.com/Mintplex-Labs/anything-llm) is an open-source platform and desktop app that is an "all-in-one" solution for using LLMs to their maximum potential, but without all the complexity or setup. + +[AnythingLLM Hub](https://hub.anythingllm.com) is a platform for sharing and discovering custom community-contributed plugins for AnythingLLM. + +AnythingLLM plugins extend the functionality of AnythingLLM beyond the core set of plugins that are distributed with AnythingLLM so that you can build custom workflows and integrations or just tools that are specific to your use case. + +This CLI tool allows you to upload your custom plugins to the AnythingLLM Hub so that they can be discovered and used by others. + +### What is a plugin? + +Plugins are extensions to AnythingLLM that can be used to add custom functionality to AnythingLLM. These are pieces of custom code that can do whatever it is you want them to do. Anything you can write in Node.js can be used as a plugin within the framework of AnythingLLM. + +### Supported Entity Types + +- [x] Agent Skills +- [ ] Data Connectors + + +## Usage + +- `npx anythingllm-hub-cli help` - Show help for the CLI +- `npx anythingllm-hub-cli config` - Show the current config path and display the current config +- `npx anythingllm-hub-cli login` - Login using your [AnythingLLM Hub Connection Key](https://hub.anythingllm.com/me) +- `npx anythingllm-hub-cli logout` - Logout and clear your connection key from the config file +- `npx anythingllm-hub-cli upload --type= --path=` - Upload an agent skill to the AnythingLLM Hub + +Available entity types: +- `agent-skill` - An agent skill is a plugin that can be used to add custom functionality to AnythingLLM. + + + + +## Development +Link the local package so you can test it out without publishing: +```bash +npm link +``` + +Test it out by running `anythingllm-hub-cli` in your terminal. + +`npm unlink` to remove the link when you're done. diff --git a/bin.js b/bin.js new file mode 100755 index 0000000..9dc80ab --- /dev/null +++ b/bin.js @@ -0,0 +1,112 @@ +#!/usr/bin/env node + + +console.log('Node version:', process.version); +console.log('Node path:', process.execPath); + + +const yargs = require('yargs/yargs') +const { hideBin } = require('yargs/helpers') +const argv = yargs(hideBin(process.argv)).argv +const _AnythingLLMStore = require('./AnythingLLMStore'); + + +/** @type {_AnythingLLMStore} */ +let LocalStore; + +const Commands = { + help: { + args: {}, + description: 'Show help for the CLI', + handler: () => { + console.log('Usage: anythingllm-hub-cli [options]'); + Object.keys(Commands).forEach(command => { + const commandInfo = Commands[command]; + let info = `\t${command}: ${commandInfo.description}`; + if (commandInfo.args) { + Object.keys(commandInfo.args).forEach(arg => { + info += `\n\t\t--${arg}: ${commandInfo.args[arg].description}.${commandInfo.args[arg].choices ? `\n\t\t\tChoices: ${commandInfo.args[arg].choices.join(', ')}` : ''}`; + }); + } + console.log(info); + }); + } + }, + config: { + args: {}, + description: 'Show the current config', + handler: () => { + LocalStore.showConfig(); + } + }, + login: { + args: {}, + description: 'Login to the AnythingLLM Hub via Connection Key', + handler: async () => { + await LocalStore.login(); + } + }, + logout: { + args: {}, + description: 'Logout from the AnythingLLM Hub & clears the connection key from the config file', + handler: async () => { + await LocalStore.logout(); + } + }, + upload: { + args: { + type: { + type: 'string', + description: 'The type of item to upload', + choices: ['agent-skill'] + }, + path: { + type: 'string', + description: 'The path to the folder to upload, can be relative to the current working directory', + } + }, + description: 'Upload a model to the AnythingLLM Hub', + handler: async (args) => { + await LocalStore.upload(args); + } + }, + init: { + args: { + type: { + type: 'string', + description: 'The type of item to create', + choices: ['agent-skill'] + }, + output: { + type: 'string', + description: 'The path to save the new item to, can be relative to the current working directory', + } + }, + description: 'Initialize a new agent skill', + handler: async (args) => { + await LocalStore.newItem(args); + } + } +} + +async function main() { + const command = argv?._[0] ?? null; + if (command) { + const argObject = Object.entries(Commands[command].args).reduce((acc, [arg]) => { + if (['_', '$0'].includes(arg)) return acc; + acc[arg] = argv[arg]; + return acc; + }, {}); + LocalStore = new _AnythingLLMStore({ debug: argv.debug }); + await Commands[command].handler(argObject); + } else { + Commands.help.handler(); + } + process.exit(0); +} + +main().catch(error => { + console.error(error); + console.error('Error:', error.message); + process.exit(1); +}); diff --git a/package.json b/package.json new file mode 100644 index 0000000..7b8f6af --- /dev/null +++ b/package.json @@ -0,0 +1,46 @@ +{ + "name": "@mintplexlabs/anythingllm-hub-cli", + "version": "0.0.1", + "description": "CLI tool for interacting with the AnythingLLM Hub for custom agent skills and data connectors", + "engines": { + "node": ">=16.0.0" + }, + "engineStrict": true, + "main": "bin.js", + "bin": { + "anythingllm-hub-cli": "./bin.js" + }, + "files": [ + "bin.js", + "AnythingLLMStore.js", + "CommunityHubInterface.js", + "utils/*.js" + ], + "scripts": { + "test": "echo \"No tests specified\"" + }, + "keywords": [ + "cli", + "anythingllm", + "hub", + "agent", + "skills" + ], + "author": "Mintplex Labs", + "license": "MIT", + "dependencies": { + "acorn": "^8.14.0", + "archiver": "^7.0.1", + "input": "^1.0.1", + "node-fetch": "2", + "yargs": "^17.7.2" + }, + "repository": { + "type": "git", + "url": "git+https://github.com/mintplexlabs/anythingllm-hub-cli.git" + }, + "bugs": { + "url": "https://github.com/mintplexlabs/anythingllm-hub-cli/issues" + }, + "homepage": "https://github.com/mintplexlabs/anythingllm-hub-cli#readme" +} \ No newline at end of file diff --git a/template/agent-skill/handler.js b/template/agent-skill/handler.js new file mode 100644 index 0000000..fad36a5 --- /dev/null +++ b/template/agent-skill/handler.js @@ -0,0 +1,40 @@ +// Documentation: https://docs.anythingllm.com/agent/custom/introduction + +/** + * @typedef {Object} AnythingLLM + * @property {import('./plugin.json')} config - your plugin's config + * @property {function(string|Error): void} logger - Logging function + * @property {function(string): void} introspect - Print a string to the UI while agent skill is running + */ + +/** @type {AnythingLLM} */ +module.exports.runtime = { + /** + * @param {import('./plugin.json')['entrypoint']['params']} args - Arguments passed to the agent skill - defined in plugin.json + */ + handler: async function (args = {}) { + const callerId = `${this.config.name}-v${this.config.version}`; + try { + // Developer note: + // 1. You can use the args object to process in any data you need to the agent skill - this is coming from the LLM when the agent skill is called + // 2. You should _ALWAYS_ return a string from the handler function. This is the output passed back to the LLM. + // 3. Agents execute in a NodeJS environment so you have access to all the built in Node modules + // 4. If you require a library, then you need to manually include it as an import at the top of the file or via local node_modules folder in the plugin folder + // this functionality is experimental and will include the entire node_modules folder in the plugin zip file. + + // Do something here! + // Fetch data from an API + // Do some processing + // **if on desktop** access OS level APIs + // ... + + return "Agent skill executed successfully."; + } catch (e) { + this.logger(e) + this.introspect( + `${callerId} failed to execute. Reason: ${e.message}` + ); + return `Failed to execute agent skill. Error ${e.message}`; + } + } +}; \ No newline at end of file diff --git a/template/agent-skill/plugin.json b/template/agent-skill/plugin.json new file mode 100644 index 0000000..4029a27 --- /dev/null +++ b/template/agent-skill/plugin.json @@ -0,0 +1,19 @@ +{ + "name": "My Agent Skill", + "version": "1.0.0", + "description": "The template agent skill", + "author": null, + "license": "MIT", + "active": false, + "hubId": "my-agent-skill", + "examples": [ + { + "prompt": "Do something", + "call": "{}" + } + ], + "entrypoint": { + "file": "handler.js", + "params": {} + } +} \ No newline at end of file diff --git a/template/agent-skill/readme.md b/template/agent-skill/readme.md new file mode 100644 index 0000000..1c50498 --- /dev/null +++ b/template/agent-skill/readme.md @@ -0,0 +1,11 @@ +## My Agent Skill + +This is a template agent skill for AnythingLLM. + +## Setup + +Describe how to setup the agent skill if any post-installation steps are required + +## Usage + +Describe how to use the agent skill once it's installed, including any examples or prompts that are useful to know. diff --git a/utils/validateAgentSkill.js b/utils/validateAgentSkill.js new file mode 100644 index 0000000..beaf433 --- /dev/null +++ b/utils/validateAgentSkill.js @@ -0,0 +1,256 @@ +const fs = require('fs'); +const path = require('path'); +const acorn = require('acorn'); +const input = require('input'); +const archiver = require('archiver'); +const ALLOWED_EXTENSIONS = ['js', 'json', 'txt', 'md']; +const PLUGIN_JSON_FIELDS = { + 'hubId': { + default: null, + validate: function () { + return this.default; + } + }, + 'active': { + default: false, + validate: function () { + return this.default; + } + }, + 'name': { + default: null, + validate: async function ({ name }) { + if (!name) { + const newName = await input.text('Please enter a name for the skill:'); + if (!newName) throw new Error('Name is required'); + return newName; + } + return name; + } + }, + 'version': { + default: '0.0.1', + validate: function ({ version }) { + if (!version) return this.default; + return version; + } + }, + 'description': { + default: null, + validate: async function ({ description }) { + if (!description) { + const newDescription = await input.text('Please enter a description for the skill:'); + if (!newDescription) throw new Error('Description is required'); + return newDescription; + } + return description; + } + }, + 'author': { + default: null, + validate: function ({ author }) { + if (!author) return this.default; + return author; + } + }, + 'author_url': { + default: null, + validate: function ({ author_url }) { + if (!author_url) return this.default; + return author_url; + } + }, + 'license': { + default: null, + validate: async function ({ license }) { + if (!license) { + const newLicense = await input.select('Please select a license for the skill:', ['MIT', 'Apache-2.0', 'GPL-3.0', 'BSD-3-Clause', 'Other']); + if (!newLicense) throw new Error('License is required'); + return newLicense; + } + return license; + } + }, + 'examples': { + default: [], + validate: async function ({ examples }) { + if (!examples || !examples.length) { + const skip = await input.confirm('No examples provided - providing examples is highly recommended and will help LLMs understand how to use the skill. Continue without examples?'); + if (skip) return this.default; + throw new Error('Exited by user interrupt'); + } + return examples; + } + }, + 'entrypoint': { + default: { + file: 'handler.js', + params: {} + }, + validate: function ({ entrypoint }) { + if (!entrypoint) return this.default; + if (entrypoint.file !== this.default.file) throw new Error('Entrypoint file must be handler.js'); + if (entrypoint.params && typeof entrypoint.params !== 'object') throw new Error('Entrypoint params must be an object'); + return { file: entrypoint.file, params: entrypoint.params || {} }; + } + } +}; + +function createArchive(sourceDir, outPath) { + const archive = archiver('zip', { zlib: { level: 9 } }); + const stream = fs.createWriteStream(outPath); + + return new Promise((resolve, reject) => { + archive + .directory(sourceDir, false) + .on('error', err => reject(err)) + .pipe(stream) + ; + + stream.on('close', () => resolve()); + archive.finalize(); + }); +} + + +/** + * Validates an agent skill. + * @param {import('../AnythingLLMStore')} store - The AnythingLLMStore instance. + * @param {string} pathToSkillFolder - The path to the skill folder (resolved to an absolute path) + * @returns {Promise} + */ +async function validateAgentSkill(store, agentSkillFolder) { + const validFiles = await validateFiles(store, agentSkillFolder); + console.log(`Found ${validFiles.length} valid files in current directory`); + const pluginData = validFiles.find(file => file.name === 'plugin.json').content; + + console.log(`-------------------------------- +Agent Skill: ${pluginData.name} v${pluginData.version} by @${pluginData.author} +Contains ${validFiles.length} files. +-------------------------------- +`) + const upload = await input.confirm('Would you like to upload this skill to the AnythingLLM Hub?'); + if (!upload) throw new Error('Exited by user interrupt'); + + console.log('Registering skill with AnythingLLM Hub...'); + const visibility = await input.select('Do you want this skill to be visible to all users (public), or only you and teams you share it with (private)?', ['public', 'private']); + + const uploadResponse = await store.prepareUpload('agent-skill', visibility); + if (uploadResponse.error) throw new Error(uploadResponse.error); + console.log('Acquired registration ID:', uploadResponse.entityId); + + // Update plugin.json with the registration ID and set active to false + // and prepare for upload by stringifying the object + pluginData.hubId = uploadResponse.entityId; + pluginData.active = false; + validFiles.find(file => file.name === 'plugin.json').content = JSON.stringify(pluginData, null, 2); + + console.log('Creating build directory'); + let outputPath; + const tempDir = path.join(process.cwd(), uploadResponse.entityId); + try { + fs.mkdirSync(tempDir); + for (const file of validFiles) { + console.log(`>>> Copying ${file.name}...`); + fs.writeFileSync(path.join(tempDir, file.name), file.content, { encoding: 'utf8' }); + } + + outputPath = path.join(process.cwd(), `${uploadResponse.entityId}.zip`); + console.log('Creating archive...'); + await createArchive(tempDir, outputPath); + console.log('Uploading archive...'); + const ok = await store.uploadFile(uploadResponse, outputPath); + if (!ok) throw new Error('Failed to upload archive'); + + console.log('Finalizing the upload...'); + const finalizeResponse = await store.finalizeUpload( + 'agent-skill', + uploadResponse.entityId, + { + name: pluginData.name, + description: pluginData.description, + files: validFiles + } + ); + if (!finalizeResponse) throw new Error('Failed to finalize the upload'); + console.log(`🎉 Agent skill uploaded! You can find your skill at https://hub.anythingllm.com/i/agent-skill/${uploadResponse.entityId}`); + } catch (error) { + console.error('Error creating build directory:', error); + throw new Error(`Error creating build directory: ${error.message}`); + } finally { + // Delete the temp directory + if (fs.existsSync(tempDir)) fs.rmSync(tempDir, { recursive: true }); + if (!!outputPath && fs.existsSync(outputPath)) fs.rmSync(outputPath); + } +} + +async function validateFiles(store, agentSkillFolder) { + const validFiles = []; + const parentFolder = path.basename(agentSkillFolder); + const requiredFiles = ['handler.js', 'plugin.json']; + const missingFiles = requiredFiles.filter(file => !fs.existsSync(path.join(agentSkillFolder, file))); + if (missingFiles.length > 0) throw new Error(`The following required files are missing from the agent skill folder: ${missingFiles.join(', ')}`); + + for (const file of requiredFiles) { + const content = fs.readFileSync(path.join(agentSkillFolder, file), 'utf8'); + if (file === 'handler.js') { + try { + acorn.parse(content, { ecmaVersion: 2020, allowReturnOutsideFunction: false, allowHashBang: false, allowAwaitOutsideFunction: false }); + validFiles.push({ + content, + path: path.join(agentSkillFolder, file).replace(process.cwd(), parentFolder), + size: fs.statSync(path.join(agentSkillFolder, file)).size, + name: file + }); + console.log(`✅ ${file} is valid.`); + } catch (error) { + throw new Error(`Error parsing ${file}: ${error.message}`); + } + } + + if (file === 'plugin.json') { + try { + const pluginData = {} + const fileData = JSON.parse(content); + for (const key in PLUGIN_JSON_FIELDS) { + pluginData[key] = await PLUGIN_JSON_FIELDS[key].validate({ + ...fileData, + author: store.userInfo.username, + author_url: store.userInfo.author_url + }); + } + + validFiles.push({ + content: pluginData, + path: path.join(agentSkillFolder, file).replace(process.cwd(), parentFolder), + size: fs.statSync(path.join(agentSkillFolder, file)).size, + name: file + }); + console.log(`✅ ${file} is valid.`); + } catch (error) { + throw new Error(`Error parsing ${file}: ${error.message}`); + } + } + } + + fs.readdirSync(agentSkillFolder).forEach(file => { + if (requiredFiles.includes(file) || fs.lstatSync(path.join(agentSkillFolder, file)).isDirectory()) return; + + const extension = path.extname(file).slice(1); + if (!ALLOWED_EXTENSIONS.includes(extension)) { + console.warn(`⚠️ ${file} has an unsupported file extension: ${extension} - it will be ignored`); + } else { + console.log(`✅ ${file} is valid.`); + validFiles.push({ + content: fs.readFileSync(path.join(agentSkillFolder, file), 'utf8'), + path: path.join(agentSkillFolder, file).replace(process.cwd(), parentFolder), + size: fs.statSync(path.join(agentSkillFolder, file)).size, + name: file + }); + } + }); + + return validFiles; +} + +module.exports = validateAgentSkill; diff --git a/yarn.lock b/yarn.lock new file mode 100644 index 0000000..e067125 --- /dev/null +++ b/yarn.lock @@ -0,0 +1,792 @@ +# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. +# yarn lockfile v1 + + +"@isaacs/cliui@^8.0.2": + version "8.0.2" + resolved "https://registry.yarnpkg.com/@isaacs/cliui/-/cliui-8.0.2.tgz#b37667b7bc181c168782259bab42474fbf52b550" + integrity sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA== + dependencies: + string-width "^5.1.2" + string-width-cjs "npm:string-width@^4.2.0" + strip-ansi "^7.0.1" + strip-ansi-cjs "npm:strip-ansi@^6.0.1" + wrap-ansi "^8.1.0" + wrap-ansi-cjs "npm:wrap-ansi@^7.0.0" + +"@pkgjs/parseargs@^0.11.0": + version "0.11.0" + resolved "https://registry.yarnpkg.com/@pkgjs/parseargs/-/parseargs-0.11.0.tgz#a77ea742fab25775145434eb1d2328cf5013ac33" + integrity sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg== + +abort-controller@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/abort-controller/-/abort-controller-3.0.0.tgz#eaf54d53b62bae4138e809ca225c8439a6efb392" + integrity sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg== + dependencies: + event-target-shim "^5.0.0" + +acorn@^8.14.0: + version "8.14.0" + resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.14.0.tgz#063e2c70cac5fb4f6467f0b11152e04c682795b0" + integrity sha512-cl669nCJTZBsL97OF4kUQm5g5hC2uihk0NxY3WENAC0TYdILVkAyHymAntgxGkl7K+t0cXIrH5siy5S4XkFycA== + +ansi-escapes@^1.1.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/ansi-escapes/-/ansi-escapes-1.4.0.tgz#d3a8a83b319aa67793662b13e761c7911422306e" + integrity sha512-wiXutNjDUlNEDWHcYH3jtZUhd3c4/VojassD8zHdHCY13xbZy2XbW+NKQwA0tWGBVzDA9qEzYwfoSsWmviidhw== + +ansi-regex@^2.0.0: + version "2.1.1" + resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-2.1.1.tgz#c3b33ab5ee360d86e0e628f0468ae7ef27d654df" + integrity sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA== + +ansi-regex@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.1.tgz#082cb2c89c9fe8659a311a53bd6a4dc5301db304" + integrity sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ== + +ansi-regex@^6.0.1: + version "6.1.0" + resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-6.1.0.tgz#95ec409c69619d6cb1b8b34f14b660ef28ebd654" + integrity sha512-7HSX4QQb4CspciLpVFwyRe79O3xsIZDDLER21kERQ71oaPodF8jL725AgJMFAYbooIqolJoRLuM81SpeUkpkvA== + +ansi-styles@^2.2.1: + version "2.2.1" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-2.2.1.tgz#b432dd3358b634cf75e1e4664368240533c1ddbe" + integrity sha512-kmCevFghRiWM7HB5zTPULl4r9bVFSWjz62MhqizDGUrq2NWuNMQyuv4tHHoKJHs69M/MF64lEcHdYIocrdWQYA== + +ansi-styles@^4.0.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz#edd803628ae71c04c85ae7a0906edad34b648937" + integrity sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg== + dependencies: + color-convert "^2.0.1" + +ansi-styles@^6.1.0: + version "6.2.1" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-6.2.1.tgz#0e62320cf99c21afff3b3012192546aacbfb05c5" + integrity sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug== + +archiver-utils@^5.0.0, archiver-utils@^5.0.2: + version "5.0.2" + resolved "https://registry.yarnpkg.com/archiver-utils/-/archiver-utils-5.0.2.tgz#63bc719d951803efc72cf961a56ef810760dd14d" + integrity sha512-wuLJMmIBQYCsGZgYLTy5FIB2pF6Lfb6cXMSF8Qywwk3t20zWnAi7zLcQFdKQmIB8wyZpY5ER38x08GbwtR2cLA== + dependencies: + glob "^10.0.0" + graceful-fs "^4.2.0" + is-stream "^2.0.1" + lazystream "^1.0.0" + lodash "^4.17.15" + normalize-path "^3.0.0" + readable-stream "^4.0.0" + +archiver@^7.0.1: + version "7.0.1" + resolved "https://registry.yarnpkg.com/archiver/-/archiver-7.0.1.tgz#c9d91c350362040b8927379c7aa69c0655122f61" + integrity sha512-ZcbTaIqJOfCc03QwD468Unz/5Ir8ATtvAHsK+FdXbDIbGfihqh9mrvdcYunQzqn4HrvWWaFyaxJhGZagaJJpPQ== + dependencies: + archiver-utils "^5.0.2" + async "^3.2.4" + buffer-crc32 "^1.0.0" + readable-stream "^4.0.0" + readdir-glob "^1.1.2" + tar-stream "^3.0.0" + zip-stream "^6.0.1" + +async@^3.2.4: + version "3.2.6" + resolved "https://registry.yarnpkg.com/async/-/async-3.2.6.tgz#1b0728e14929d51b85b449b7f06e27c1145e38ce" + integrity sha512-htCUDlxyyCLMgaM3xXg0C0LW2xqfuQ6p05pCEIsXuyQ+a1koYKTuBMzRNwmybfLgvJDMd0r1LTn4+E0Ti6C2AA== + +b4a@^1.6.4: + version "1.6.7" + resolved "https://registry.yarnpkg.com/b4a/-/b4a-1.6.7.tgz#a99587d4ebbfbd5a6e3b21bdb5d5fa385767abe4" + integrity sha512-OnAYlL5b7LEkALw87fUVafQw5rVR9RjwGd4KUwNQ6DrrNmaVaUCgLipfVlzrPQ4tWOR9P0IXGNOx50jYCCdSJg== + +babel-runtime@^6.6.1: + version "6.26.0" + resolved "https://registry.yarnpkg.com/babel-runtime/-/babel-runtime-6.26.0.tgz#965c7058668e82b55d7bfe04ff2337bc8b5647fe" + integrity sha512-ITKNuq2wKlW1fJg9sSW52eepoYgZBggvOAHC0u/CYu/qxQ9EVzThCgR69BnSXLHjy2f7SY5zaQ4yt7H9ZVxY2g== + dependencies: + core-js "^2.4.0" + regenerator-runtime "^0.11.0" + +balanced-match@^1.0.0: + version "1.0.2" + resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee" + integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw== + +bare-events@^2.2.0: + version "2.5.0" + resolved "https://registry.yarnpkg.com/bare-events/-/bare-events-2.5.0.tgz#305b511e262ffd8b9d5616b056464f8e1b3329cc" + integrity sha512-/E8dDe9dsbLyh2qrZ64PEPadOQ0F4gbl1sUJOrmph7xOiIxfY8vwab/4bFLh4Y88/Hk/ujKcrQKc+ps0mv873A== + +base64-js@^1.3.1: + version "1.5.1" + resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a" + integrity sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA== + +brace-expansion@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-2.0.1.tgz#1edc459e0f0c548486ecf9fc99f2221364b9a0ae" + integrity sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA== + dependencies: + balanced-match "^1.0.0" + +buffer-crc32@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/buffer-crc32/-/buffer-crc32-1.0.0.tgz#a10993b9055081d55304bd9feb4a072de179f405" + integrity sha512-Db1SbgBS/fg/392AblrMJk97KggmvYhr4pB5ZIMTWtaivCPMWLkmb7m21cJvpvgK+J3nsU2CmmixNBZx4vFj/w== + +buffer@^6.0.3: + version "6.0.3" + resolved "https://registry.yarnpkg.com/buffer/-/buffer-6.0.3.tgz#2ace578459cc8fbe2a70aaa8f52ee63b6a74c6c6" + integrity sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA== + dependencies: + base64-js "^1.3.1" + ieee754 "^1.2.1" + +chalk@^1.0.0, chalk@^1.1.1: + version "1.1.3" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-1.1.3.tgz#a8115c55e4a702fe4d150abd3872822a7e09fc98" + integrity sha512-U3lRVLMSlsCfjqYPbLyVv11M9CPW4I728d6TCKMAOJueEeB9/8o+eSsMnxPJD+Q+K909sdESg7C+tIkoH6on1A== + dependencies: + ansi-styles "^2.2.1" + escape-string-regexp "^1.0.2" + has-ansi "^2.0.0" + strip-ansi "^3.0.0" + supports-color "^2.0.0" + +cli-cursor@^1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/cli-cursor/-/cli-cursor-1.0.2.tgz#64da3f7d56a54412e59794bd62dc35295e8f2987" + integrity sha512-25tABq090YNKkF6JH7lcwO0zFJTRke4Jcq9iX2nr/Sz0Cjjv4gckmwlW6Ty/aoyFd6z3ysR2hMGC2GFugmBo6A== + dependencies: + restore-cursor "^1.0.1" + +cli-width@^2.0.0: + version "2.2.1" + resolved "https://registry.yarnpkg.com/cli-width/-/cli-width-2.2.1.tgz#b0433d0b4e9c847ef18868a4ef16fd5fc8271c48" + integrity sha512-GRMWDxpOB6Dgk2E5Uo+3eEBvtOOlimMmpbFiKuLFnQzYDavtLFY3K5ona41jgN/WdRZtG7utuVSVTL4HbZHGkw== + +cliui@^8.0.1: + version "8.0.1" + resolved "https://registry.yarnpkg.com/cliui/-/cliui-8.0.1.tgz#0c04b075db02cbfe60dc8e6cf2f5486b1a3608aa" + integrity sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ== + dependencies: + string-width "^4.2.0" + strip-ansi "^6.0.1" + wrap-ansi "^7.0.0" + +code-point-at@^1.0.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/code-point-at/-/code-point-at-1.1.0.tgz#0d070b4d043a5bea33a2f1a40e2edb3d9a4ccf77" + integrity sha512-RpAVKQA5T63xEj6/giIbUEtZwJ4UFIc3ZtvEkiaUERylqe8xb5IvqcgOurZLahv93CLKfxcw5YI+DZcUBRyLXA== + +color-convert@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3" + integrity sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ== + dependencies: + color-name "~1.1.4" + +color-name@~1.1.4: + version "1.1.4" + resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2" + integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== + +compress-commons@^6.0.2: + version "6.0.2" + resolved "https://registry.yarnpkg.com/compress-commons/-/compress-commons-6.0.2.tgz#26d31251a66b9d6ba23a84064ecd3a6a71d2609e" + integrity sha512-6FqVXeETqWPoGcfzrXb37E50NP0LXT8kAMu5ooZayhWWdgEY4lBEEcbQNXtkuKQsGduxiIcI4gOTsxTmuq/bSg== + dependencies: + crc-32 "^1.2.0" + crc32-stream "^6.0.0" + is-stream "^2.0.1" + normalize-path "^3.0.0" + readable-stream "^4.0.0" + +core-js@^2.4.0: + version "2.6.12" + resolved "https://registry.yarnpkg.com/core-js/-/core-js-2.6.12.tgz#d9333dfa7b065e347cc5682219d6f690859cc2ec" + integrity sha512-Kb2wC0fvsWfQrgk8HU5lW6U/Lcs8+9aaYcy4ZFc6DDlo4nZ7n70dEgE5rtR0oG6ufKDUnrwfWL1mXR5ljDatrQ== + +core-util-is@~1.0.0: + version "1.0.3" + resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.3.tgz#a6042d3634c2b27e9328f837b965fac83808db85" + integrity sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ== + +crc-32@^1.2.0: + version "1.2.2" + resolved "https://registry.yarnpkg.com/crc-32/-/crc-32-1.2.2.tgz#3cad35a934b8bf71f25ca524b6da51fb7eace2ff" + integrity sha512-ROmzCKrTnOwybPcJApAA6WBWij23HVfGVNKqqrZpuyZOHqK2CwHSvpGuyt/UNNvaIjEd8X5IFGp4Mh+Ie1IHJQ== + +crc32-stream@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/crc32-stream/-/crc32-stream-6.0.0.tgz#8529a3868f8b27abb915f6c3617c0fadedbf9430" + integrity sha512-piICUB6ei4IlTv1+653yq5+KoqfBYmj9bw6LqXoOneTMDXk5nM1qt12mFW1caG3LlJXEKW1Bp0WggEmIfQB34g== + dependencies: + crc-32 "^1.2.0" + readable-stream "^4.0.0" + +cross-spawn@^7.0.0: + version "7.0.6" + resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-7.0.6.tgz#8a58fe78f00dcd70c370451759dfbfaf03e8ee9f" + integrity sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA== + dependencies: + path-key "^3.1.0" + shebang-command "^2.0.0" + which "^2.0.1" + +eastasianwidth@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/eastasianwidth/-/eastasianwidth-0.2.0.tgz#696ce2ec0aa0e6ea93a397ffcf24aa7840c827cb" + integrity sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA== + +emoji-regex@^8.0.0: + version "8.0.0" + resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37" + integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A== + +emoji-regex@^9.2.2: + version "9.2.2" + resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-9.2.2.tgz#840c8803b0d8047f4ff0cf963176b32d4ef3ed72" + integrity sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg== + +escalade@^3.1.1: + version "3.2.0" + resolved "https://registry.yarnpkg.com/escalade/-/escalade-3.2.0.tgz#011a3f69856ba189dffa7dc8fcce99d2a87903e5" + integrity sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA== + +escape-string-regexp@^1.0.2, escape-string-regexp@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4" + integrity sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg== + +event-target-shim@^5.0.0: + version "5.0.1" + resolved "https://registry.yarnpkg.com/event-target-shim/-/event-target-shim-5.0.1.tgz#5d4d3ebdf9583d63a5333ce2deb7480ab2b05789" + integrity sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ== + +events@^3.3.0: + version "3.3.0" + resolved "https://registry.yarnpkg.com/events/-/events-3.3.0.tgz#31a95ad0a924e2d2c419a813aeb2c4e878ea7400" + integrity sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q== + +exit-hook@^1.0.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/exit-hook/-/exit-hook-1.1.1.tgz#f05ca233b48c05d54fff07765df8507e95c02ff8" + integrity sha512-MsG3prOVw1WtLXAZbM3KiYtooKR1LvxHh3VHsVtIy0uiUu8usxgB/94DP2HxtD/661lLdB6yzQ09lGJSQr6nkg== + +fast-fifo@^1.2.0, fast-fifo@^1.3.2: + version "1.3.2" + resolved "https://registry.yarnpkg.com/fast-fifo/-/fast-fifo-1.3.2.tgz#286e31de96eb96d38a97899815740ba2a4f3640c" + integrity sha512-/d9sfos4yxzpwkDkuN7k2SqFKtYNmCTzgfEpz82x34IM9/zc8KGxQoXg1liNC/izpRM/MBdt44Nmx41ZWqk+FQ== + +figures@^1.3.5: + version "1.7.0" + resolved "https://registry.yarnpkg.com/figures/-/figures-1.7.0.tgz#cbe1e3affcf1cd44b80cadfed28dc793a9701d2e" + integrity sha512-UxKlfCRuCBxSXU4C6t9scbDyWZ4VlaFFdojKtzJuSkuOBQ5CNFum+zZXFwHjo+CxBC1t6zlYPgHIgFjL8ggoEQ== + dependencies: + escape-string-regexp "^1.0.5" + object-assign "^4.1.0" + +foreground-child@^3.1.0: + version "3.3.0" + resolved "https://registry.yarnpkg.com/foreground-child/-/foreground-child-3.3.0.tgz#0ac8644c06e431439f8561db8ecf29a7b5519c77" + integrity sha512-Ld2g8rrAyMYFXBhEqMz8ZAHBi4J4uS1i/CxGMDnjyFWddMXLVcDp051DZfu+t7+ab7Wv6SMqpWmyFIj5UbfFvg== + dependencies: + cross-spawn "^7.0.0" + signal-exit "^4.0.1" + +get-caller-file@^2.0.5: + version "2.0.5" + resolved "https://registry.yarnpkg.com/get-caller-file/-/get-caller-file-2.0.5.tgz#4f94412a82db32f36e3b0b9741f8a97feb031f7e" + integrity sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg== + +glob@^10.0.0: + version "10.4.5" + resolved "https://registry.yarnpkg.com/glob/-/glob-10.4.5.tgz#f4d9f0b90ffdbab09c9d77f5f29b4262517b0956" + integrity sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg== + dependencies: + foreground-child "^3.1.0" + jackspeak "^3.1.2" + minimatch "^9.0.4" + minipass "^7.1.2" + package-json-from-dist "^1.0.0" + path-scurry "^1.11.1" + +graceful-fs@^4.2.0: + version "4.2.11" + resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.11.tgz#4183e4e8bf08bb6e05bbb2f7d2e0c8f712ca40e3" + integrity sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ== + +has-ansi@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/has-ansi/-/has-ansi-2.0.0.tgz#34f5049ce1ecdf2b0649af3ef24e45ed35416d91" + integrity sha512-C8vBJ8DwUCx19vhm7urhTuUsr4/IyP6l4VzNQDv+ryHQObW3TTTp9yB68WpYgRe2bbaGuZ/se74IqFeVnMnLZg== + dependencies: + ansi-regex "^2.0.0" + +ieee754@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.2.1.tgz#8eb7a10a63fff25d15a57b001586d177d1b0d352" + integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA== + +inherits@~2.0.3: + version "2.0.4" + resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c" + integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== + +input@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/input/-/input-1.0.1.tgz#26bfb5315959c403e9317175cf8f9894e97e5742" + integrity sha512-5DKQKQ7Nm/CaPGYKF74uUvk5ftC3S04fLYWcDrNG2rOVhhRgB4E2J8JNb7AAh+RlQ/954ukas4bEbrRQ3/kPGA== + dependencies: + babel-runtime "^6.6.1" + chalk "^1.1.1" + inquirer "^0.12.0" + lodash "^4.6.1" + +inquirer@^0.12.0: + version "0.12.0" + resolved "https://registry.yarnpkg.com/inquirer/-/inquirer-0.12.0.tgz#1ef2bfd63504df0bc75785fff8c2c41df12f077e" + integrity sha512-bOetEz5+/WpgaW4D1NYOk1aD+JCqRjqu/FwRFgnIfiP7FC/zinsrfyO1vlS3nyH/R7S0IH3BIHBu4DBIDSqiGQ== + dependencies: + ansi-escapes "^1.1.0" + ansi-regex "^2.0.0" + chalk "^1.0.0" + cli-cursor "^1.0.1" + cli-width "^2.0.0" + figures "^1.3.5" + lodash "^4.3.0" + readline2 "^1.0.1" + run-async "^0.1.0" + rx-lite "^3.1.2" + string-width "^1.0.1" + strip-ansi "^3.0.0" + through "^2.3.6" + +is-fullwidth-code-point@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz#ef9e31386f031a7f0d643af82fde50c457ef00cb" + integrity sha512-1pqUqRjkhPJ9miNq9SwMfdvi6lBJcd6eFxvfaivQhaH3SgisfiuudvFntdKOmxuee/77l+FPjKrQjWvmPjWrRw== + dependencies: + number-is-nan "^1.0.0" + +is-fullwidth-code-point@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz#f116f8064fe90b3f7844a38997c0b75051269f1d" + integrity sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg== + +is-stream@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/is-stream/-/is-stream-2.0.1.tgz#fac1e3d53b97ad5a9d0ae9cef2389f5810a5c077" + integrity sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg== + +isarray@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11" + integrity sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ== + +isexe@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" + integrity sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw== + +jackspeak@^3.1.2: + version "3.4.3" + resolved "https://registry.yarnpkg.com/jackspeak/-/jackspeak-3.4.3.tgz#8833a9d89ab4acde6188942bd1c53b6390ed5a8a" + integrity sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw== + dependencies: + "@isaacs/cliui" "^8.0.2" + optionalDependencies: + "@pkgjs/parseargs" "^0.11.0" + +lazystream@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/lazystream/-/lazystream-1.0.1.tgz#494c831062f1f9408251ec44db1cba29242a2638" + integrity sha512-b94GiNHQNy6JNTrt5w6zNyffMrNkXZb3KTkCZJb2V1xaEGCk093vkZ2jk3tpaeP33/OiXC+WvK9AxUebnf5nbw== + dependencies: + readable-stream "^2.0.5" + +lodash@^4.17.15, lodash@^4.3.0, lodash@^4.6.1: + version "4.17.21" + resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c" + integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg== + +lru-cache@^10.2.0: + version "10.4.3" + resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-10.4.3.tgz#410fc8a17b70e598013df257c2446b7f3383f119" + integrity sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ== + +minimatch@^5.1.0: + version "5.1.6" + resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-5.1.6.tgz#1cfcb8cf5522ea69952cd2af95ae09477f122a96" + integrity sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g== + dependencies: + brace-expansion "^2.0.1" + +minimatch@^9.0.4: + version "9.0.5" + resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-9.0.5.tgz#d74f9dd6b57d83d8e98cfb82133b03978bc929e5" + integrity sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow== + dependencies: + brace-expansion "^2.0.1" + +"minipass@^5.0.0 || ^6.0.2 || ^7.0.0", minipass@^7.1.2: + version "7.1.2" + resolved "https://registry.yarnpkg.com/minipass/-/minipass-7.1.2.tgz#93a9626ce5e5e66bd4db86849e7515e92340a707" + integrity sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw== + +mute-stream@0.0.5: + version "0.0.5" + resolved "https://registry.yarnpkg.com/mute-stream/-/mute-stream-0.0.5.tgz#8fbfabb0a98a253d3184331f9e8deb7372fac6c0" + integrity sha512-EbrziT4s8cWPmzr47eYVW3wimS4HsvlnV5ri1xw1aR6JQo/OrJX5rkl32K/QQHdxeabJETtfeaROGhd8W7uBgg== + +node-fetch@2: + version "2.7.0" + resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-2.7.0.tgz#d0f0fa6e3e2dc1d27efcd8ad99d550bda94d187d" + integrity sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A== + dependencies: + whatwg-url "^5.0.0" + +normalize-path@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/normalize-path/-/normalize-path-3.0.0.tgz#0dcd69ff23a1c9b11fd0978316644a0388216a65" + integrity sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA== + +number-is-nan@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/number-is-nan/-/number-is-nan-1.0.1.tgz#097b602b53422a522c1afb8790318336941a011d" + integrity sha512-4jbtZXNAsfZbAHiiqjLPBiCl16dES1zI4Hpzzxw61Tk+loF+sBDBKx1ICKKKwIqQ7M0mFn1TmkN7euSncWgHiQ== + +object-assign@^4.1.0: + version "4.1.1" + resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863" + integrity sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg== + +once@^1.3.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1" + integrity sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w== + dependencies: + wrappy "1" + +onetime@^1.0.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/onetime/-/onetime-1.1.0.tgz#a1f7838f8314c516f05ecefcbc4ccfe04b4ed789" + integrity sha512-GZ+g4jayMqzCRMgB2sol7GiCLjKfS1PINkjmx8spcKce1LiVqcbQreXwqs2YAFXC6R03VIG28ZS31t8M866v6A== + +package-json-from-dist@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/package-json-from-dist/-/package-json-from-dist-1.0.1.tgz#4f1471a010827a86f94cfd9b0727e36d267de505" + integrity sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw== + +path-key@^3.1.0: + version "3.1.1" + resolved "https://registry.yarnpkg.com/path-key/-/path-key-3.1.1.tgz#581f6ade658cbba65a0d3380de7753295054f375" + integrity sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q== + +path-scurry@^1.11.1: + version "1.11.1" + resolved "https://registry.yarnpkg.com/path-scurry/-/path-scurry-1.11.1.tgz#7960a668888594a0720b12a911d1a742ab9f11d2" + integrity sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA== + dependencies: + lru-cache "^10.2.0" + minipass "^5.0.0 || ^6.0.2 || ^7.0.0" + +process-nextick-args@~2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/process-nextick-args/-/process-nextick-args-2.0.1.tgz#7820d9b16120cc55ca9ae7792680ae7dba6d7fe2" + integrity sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag== + +process@^0.11.10: + version "0.11.10" + resolved "https://registry.yarnpkg.com/process/-/process-0.11.10.tgz#7332300e840161bda3e69a1d1d91a7d4bc16f182" + integrity sha512-cdGef/drWFoydD1JsMzuFf8100nZl+GT+yacc2bEced5f9Rjk4z+WtFUTBu9PhOi9j/jfmBPu0mMEY4wIdAF8A== + +queue-tick@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/queue-tick/-/queue-tick-1.0.1.tgz#f6f07ac82c1fd60f82e098b417a80e52f1f4c142" + integrity sha512-kJt5qhMxoszgU/62PLP1CJytzd2NKetjSRnyuj31fDd3Rlcz3fzlFdFLD1SItunPwyqEOkca6GbV612BWfaBag== + +readable-stream@^2.0.5: + version "2.3.8" + resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-2.3.8.tgz#91125e8042bba1b9887f49345f6277027ce8be9b" + integrity sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA== + dependencies: + core-util-is "~1.0.0" + inherits "~2.0.3" + isarray "~1.0.0" + process-nextick-args "~2.0.0" + safe-buffer "~5.1.1" + string_decoder "~1.1.1" + util-deprecate "~1.0.1" + +readable-stream@^4.0.0: + version "4.5.2" + resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-4.5.2.tgz#9e7fc4c45099baeed934bff6eb97ba6cf2729e09" + integrity sha512-yjavECdqeZ3GLXNgRXgeQEdz9fvDDkNKyHnbHRFtOr7/LcfgBcmct7t/ET+HaCTqfh06OzoAxrkN/IfjJBVe+g== + dependencies: + abort-controller "^3.0.0" + buffer "^6.0.3" + events "^3.3.0" + process "^0.11.10" + string_decoder "^1.3.0" + +readdir-glob@^1.1.2: + version "1.1.3" + resolved "https://registry.yarnpkg.com/readdir-glob/-/readdir-glob-1.1.3.tgz#c3d831f51f5e7bfa62fa2ffbe4b508c640f09584" + integrity sha512-v05I2k7xN8zXvPD9N+z/uhXPaj0sUFCe2rcWZIpBsqxfP7xXFQ0tipAd/wjj1YxWyWtUS5IDJpOG82JKt2EAVA== + dependencies: + minimatch "^5.1.0" + +readline2@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/readline2/-/readline2-1.0.1.tgz#41059608ffc154757b715d9989d199ffbf372e35" + integrity sha512-8/td4MmwUB6PkZUbV25uKz7dfrmjYWxsW8DVfibWdlHRk/l/DfHKn4pU+dfcoGLFgWOdyGCzINRQD7jn+Bv+/g== + dependencies: + code-point-at "^1.0.0" + is-fullwidth-code-point "^1.0.0" + mute-stream "0.0.5" + +regenerator-runtime@^0.11.0: + version "0.11.1" + resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.11.1.tgz#be05ad7f9bf7d22e056f9726cee5017fbf19e2e9" + integrity sha512-MguG95oij0fC3QV3URf4V2SDYGJhJnJGqvIIgdECeODCT98wSWDAJ94SSuVpYQUoTcGUIL6L4yNB7j1DFFHSBg== + +require-directory@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/require-directory/-/require-directory-2.1.1.tgz#8c64ad5fd30dab1c976e2344ffe7f792a6a6df42" + integrity sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q== + +restore-cursor@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/restore-cursor/-/restore-cursor-1.0.1.tgz#34661f46886327fed2991479152252df92daa541" + integrity sha512-reSjH4HuiFlxlaBaFCiS6O76ZGG2ygKoSlCsipKdaZuKSPx/+bt9mULkn4l0asVzbEfQQmXRg6Wp6gv6m0wElw== + dependencies: + exit-hook "^1.0.0" + onetime "^1.0.0" + +run-async@^0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/run-async/-/run-async-0.1.0.tgz#c8ad4a5e110661e402a7d21b530e009f25f8e389" + integrity sha512-qOX+w+IxFgpUpJfkv2oGN0+ExPs68F4sZHfaRRx4dDexAQkG83atugKVEylyT5ARees3HBbfmuvnjbrd8j9Wjw== + dependencies: + once "^1.3.0" + +rx-lite@^3.1.2: + version "3.1.2" + resolved "https://registry.yarnpkg.com/rx-lite/-/rx-lite-3.1.2.tgz#19ce502ca572665f3b647b10939f97fd1615f102" + integrity sha512-1I1+G2gteLB8Tkt8YI1sJvSIfa0lWuRtC8GjvtyPBcLSF5jBCCJJqKrpER5JU5r6Bhe+i9/pK3VMuUcXu0kdwQ== + +safe-buffer@~5.1.0, safe-buffer@~5.1.1: + version "5.1.2" + resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.2.tgz#991ec69d296e0313747d59bdfd2b745c35f8828d" + integrity sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g== + +safe-buffer@~5.2.0: + version "5.2.1" + resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6" + integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ== + +shebang-command@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-2.0.0.tgz#ccd0af4f8835fbdc265b82461aaf0c36663f34ea" + integrity sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA== + dependencies: + shebang-regex "^3.0.0" + +shebang-regex@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/shebang-regex/-/shebang-regex-3.0.0.tgz#ae16f1644d873ecad843b0307b143362d4c42172" + integrity sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A== + +signal-exit@^4.0.1: + version "4.1.0" + resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-4.1.0.tgz#952188c1cbd546070e2dd20d0f41c0ae0530cb04" + integrity sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw== + +streamx@^2.15.0: + version "2.20.2" + resolved "https://registry.yarnpkg.com/streamx/-/streamx-2.20.2.tgz#6a8911959d6f307c19781a1d19ecd94b5f042d78" + integrity sha512-aDGDLU+j9tJcUdPGOaHmVF1u/hhI+CsGkT02V3OKlHDV7IukOI+nTWAGkiZEKCO35rWN1wIr4tS7YFr1f4qSvA== + dependencies: + fast-fifo "^1.3.2" + queue-tick "^1.0.1" + text-decoder "^1.1.0" + optionalDependencies: + bare-events "^2.2.0" + +"string-width-cjs@npm:string-width@^4.2.0", string-width@^4.1.0, string-width@^4.2.0, string-width@^4.2.3: + version "4.2.3" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010" + integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g== + dependencies: + emoji-regex "^8.0.0" + is-fullwidth-code-point "^3.0.0" + strip-ansi "^6.0.1" + +string-width@^1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-1.0.2.tgz#118bdf5b8cdc51a2a7e70d211e07e2b0b9b107d3" + integrity sha512-0XsVpQLnVCXHJfyEs8tC0zpTVIr5PKKsQtkT29IwupnPTjtPmQ3xT/4yCREF9hYkV/3M3kzcUTSAZT6a6h81tw== + dependencies: + code-point-at "^1.0.0" + is-fullwidth-code-point "^1.0.0" + strip-ansi "^3.0.0" + +string-width@^5.0.1, string-width@^5.1.2: + version "5.1.2" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-5.1.2.tgz#14f8daec6d81e7221d2a357e668cab73bdbca794" + integrity sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA== + dependencies: + eastasianwidth "^0.2.0" + emoji-regex "^9.2.2" + strip-ansi "^7.0.1" + +string_decoder@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.3.0.tgz#42f114594a46cf1a8e30b0a84f56c78c3edac21e" + integrity sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA== + dependencies: + safe-buffer "~5.2.0" + +string_decoder@~1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.1.1.tgz#9cf1611ba62685d7030ae9e4ba34149c3af03fc8" + integrity sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg== + dependencies: + safe-buffer "~5.1.0" + +"strip-ansi-cjs@npm:strip-ansi@^6.0.1", strip-ansi@^6.0.0, strip-ansi@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9" + integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A== + dependencies: + ansi-regex "^5.0.1" + +strip-ansi@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-3.0.1.tgz#6a385fb8853d952d5ff05d0e8aaf94278dc63dcf" + integrity sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg== + dependencies: + ansi-regex "^2.0.0" + +strip-ansi@^7.0.1: + version "7.1.0" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-7.1.0.tgz#d5b6568ca689d8561370b0707685d22434faff45" + integrity sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ== + dependencies: + ansi-regex "^6.0.1" + +supports-color@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-2.0.0.tgz#535d045ce6b6363fa40117084629995e9df324c7" + integrity sha512-KKNVtd6pCYgPIKU4cp2733HWYCpplQhddZLBUryaAHou723x+FRzQ5Df824Fj+IyyuiQTRoub4SnIFfIcrp70g== + +tar-stream@^3.0.0: + version "3.1.7" + resolved "https://registry.yarnpkg.com/tar-stream/-/tar-stream-3.1.7.tgz#24b3fb5eabada19fe7338ed6d26e5f7c482e792b" + integrity sha512-qJj60CXt7IU1Ffyc3NJMjh6EkuCFej46zUqJ4J7pqYlThyd9bO0XBTmcOIhSzZJVWfsLks0+nle/j538YAW9RQ== + dependencies: + b4a "^1.6.4" + fast-fifo "^1.2.0" + streamx "^2.15.0" + +text-decoder@^1.1.0: + version "1.2.1" + resolved "https://registry.yarnpkg.com/text-decoder/-/text-decoder-1.2.1.tgz#e173f5121d97bfa3ff8723429ad5ba92e1ead67e" + integrity sha512-x9v3H/lTKIJKQQe7RPQkLfKAnc9lUTkWDypIQgTzPJAq+5/GCDHonmshfvlsNSj58yyshbIJJDLmU15qNERrXQ== + +through@^2.3.6: + version "2.3.8" + resolved "https://registry.yarnpkg.com/through/-/through-2.3.8.tgz#0dd4c9ffaabc357960b1b724115d7e0e86a2e1f5" + integrity sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg== + +tr46@~0.0.3: + version "0.0.3" + resolved "https://registry.yarnpkg.com/tr46/-/tr46-0.0.3.tgz#8184fd347dac9cdc185992f3a6622e14b9d9ab6a" + integrity sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw== + +util-deprecate@~1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf" + integrity sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw== + +webidl-conversions@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-3.0.1.tgz#24534275e2a7bc6be7bc86611cc16ae0a5654871" + integrity sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ== + +whatwg-url@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/whatwg-url/-/whatwg-url-5.0.0.tgz#966454e8765462e37644d3626f6742ce8b70965d" + integrity sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw== + dependencies: + tr46 "~0.0.3" + webidl-conversions "^3.0.0" + +which@^2.0.1: + version "2.0.2" + resolved "https://registry.yarnpkg.com/which/-/which-2.0.2.tgz#7c6a8dd0a636a0327e10b59c9286eee93f3f51b1" + integrity sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA== + dependencies: + isexe "^2.0.0" + +"wrap-ansi-cjs@npm:wrap-ansi@^7.0.0", wrap-ansi@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43" + integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q== + dependencies: + ansi-styles "^4.0.0" + string-width "^4.1.0" + strip-ansi "^6.0.0" + +wrap-ansi@^8.1.0: + version "8.1.0" + resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-8.1.0.tgz#56dc22368ee570face1b49819975d9b9a5ead214" + integrity sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ== + dependencies: + ansi-styles "^6.1.0" + string-width "^5.0.1" + strip-ansi "^7.0.1" + +wrappy@1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" + integrity sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ== + +y18n@^5.0.5: + version "5.0.8" + resolved "https://registry.yarnpkg.com/y18n/-/y18n-5.0.8.tgz#7f4934d0f7ca8c56f95314939ddcd2dd91ce1d55" + integrity sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA== + +yargs-parser@^21.1.1: + version "21.1.1" + resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-21.1.1.tgz#9096bceebf990d21bb31fa9516e0ede294a77d35" + integrity sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw== + +yargs@^17.7.2: + version "17.7.2" + resolved "https://registry.yarnpkg.com/yargs/-/yargs-17.7.2.tgz#991df39aca675a192b816e1e0363f9d75d2aa269" + integrity sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w== + dependencies: + cliui "^8.0.1" + escalade "^3.1.1" + get-caller-file "^2.0.5" + require-directory "^2.1.1" + string-width "^4.2.3" + y18n "^5.0.5" + yargs-parser "^21.1.1" + +zip-stream@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/zip-stream/-/zip-stream-6.0.1.tgz#e141b930ed60ccaf5d7fa9c8260e0d1748a2bbfb" + integrity sha512-zK7YHHz4ZXpW89AHXUPbQVGKI7uvkd3hzusTdotCg1UxyaVtg0zFJSTfW/Dq5f7OBBVnq6cZIaC8Ti4hb6dtCA== + dependencies: + archiver-utils "^5.0.0" + compress-commons "^6.0.2" + readable-stream "^4.0.0"