From fb8bcc854d9c86e02ded788402e23ce8692b3b5c Mon Sep 17 00:00:00 2001 From: lastlink Date: Sat, 22 Jun 2024 15:45:40 -0400 Subject: [PATCH 01/12] check in current work --- .eslintignore | 3 + .eslintrc | 51 + .github/ISSUE_TEMPLATE/bug_report.md | 38 + .github/ISSUE_TEMPLATE/feature_request.md | 20 + .github/PULL_REQUEST_TEMPLATE.md | 6 + .github/dependabot.yml | 12 + .github/workflows/dependabot_automerge.yml | 14 + .github/workflows/tests.yaml | 21 + .gitignore | 1 - .npmignore | 114 + .vscode/launch.json | 53 + README.md | 3 + build/createDistFolder.js | 16 + dist/sql.js | 1459 +++++ jest.config.ts | 28 + package-lock.json | 6031 ++++++++++++++++++++ package.json | 53 + src/sql.ts | 1290 +++++ src/types/drawio-types.d.ts | 136 + src/types/sql-plugin-types.ts | 17 + tsconfig.json | 118 + 21 files changed, 9483 insertions(+), 1 deletion(-) create mode 100644 .eslintignore create mode 100644 .eslintrc create mode 100644 .github/ISSUE_TEMPLATE/bug_report.md create mode 100644 .github/ISSUE_TEMPLATE/feature_request.md create mode 100644 .github/PULL_REQUEST_TEMPLATE.md create mode 100644 .github/dependabot.yml create mode 100644 .github/workflows/dependabot_automerge.yml create mode 100644 .github/workflows/tests.yaml create mode 100644 .npmignore create mode 100644 .vscode/launch.json create mode 100644 build/createDistFolder.js create mode 100644 dist/sql.js create mode 100644 jest.config.ts create mode 100644 package-lock.json create mode 100644 package.json create mode 100644 src/sql.ts create mode 100644 src/types/drawio-types.d.ts create mode 100644 src/types/sql-plugin-types.ts create mode 100644 tsconfig.json diff --git a/.eslintignore b/.eslintignore new file mode 100644 index 0000000..a19ba71 --- /dev/null +++ b/.eslintignore @@ -0,0 +1,3 @@ +node_modules +dist +./*.js \ No newline at end of file diff --git a/.eslintrc b/.eslintrc new file mode 100644 index 0000000..0490c11 --- /dev/null +++ b/.eslintrc @@ -0,0 +1,51 @@ +{ + "parser": "@typescript-eslint/parser", + "extends": [ + "plugin:@typescript-eslint/recommended" + ], + "parserOptions": { + "ecmaVersion": 2018, + "sourceType": "module" + }, + "rules": { + "semi": [ + "error", + "always" + ], + "quotes": [ + "error", + "double" + ], + "@typescript-eslint/explicit-function-return-type": "off", + "@typescript-eslint/camelcase": "off", + "@typescript-eslint/no-explicit-any": "off", + "no-multiple-empty-lines": "error", + "@typescript-eslint/no-inferrable-types": [ + "warn", + { + "ignoreParameters": true + } + ], + "@typescript-eslint/no-unused-vars": "warn", + "@typescript-eslint/class-name-casing": "off" + }, + "ignorePatterns": [ + "src/TextHighlighter.js", + "lib/*", + "deps/*", + "dist/*", + "src/mermaid/*", + "archived" + ], + "overrides": [ + { + "files": [ + "**/__tests__/*.{j,t}s?(x)", + "**/tests/**/*.spec.{j,t}s?(x)" + ], + "env": { + "jest": true + } + } + ] +} \ No newline at end of file diff --git a/.github/ISSUE_TEMPLATE/bug_report.md b/.github/ISSUE_TEMPLATE/bug_report.md new file mode 100644 index 0000000..dd84ea7 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/bug_report.md @@ -0,0 +1,38 @@ +--- +name: Bug report +about: Create a report to help us improve +title: '' +labels: '' +assignees: '' + +--- + +**Describe the bug** +A clear and concise description of what the bug is. + +**To Reproduce** +Steps to reproduce the behavior: +1. Go to '...' +2. Click on '....' +3. Scroll down to '....' +4. See error + +**Expected behavior** +A clear and concise description of what you expected to happen. + +**Screenshots** +If applicable, add screenshots to help explain your problem. + +**Desktop (please complete the following information):** + - OS: [e.g. iOS] + - Browser [e.g. chrome, safari] + - Version [e.g. 22] + +**Smartphone (please complete the following information):** + - Device: [e.g. iPhone6] + - OS: [e.g. iOS8.1] + - Browser [e.g. stock browser, safari] + - Version [e.g. 22] + +**Additional context** +Add any other context about the problem here. diff --git a/.github/ISSUE_TEMPLATE/feature_request.md b/.github/ISSUE_TEMPLATE/feature_request.md new file mode 100644 index 0000000..bbcbbe7 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/feature_request.md @@ -0,0 +1,20 @@ +--- +name: Feature request +about: Suggest an idea for this project +title: '' +labels: '' +assignees: '' + +--- + +**Is your feature request related to a problem? Please describe.** +A clear and concise description of what the problem is. Ex. I'm always frustrated when [...] + +**Describe the solution you'd like** +A clear and concise description of what you want to happen. + +**Describe alternatives you've considered** +A clear and concise description of any alternative solutions or features you've considered. + +**Additional context** +Add any other context or screenshots about the feature request here. diff --git a/.github/PULL_REQUEST_TEMPLATE.md b/.github/PULL_REQUEST_TEMPLATE.md new file mode 100644 index 0000000..17393c8 --- /dev/null +++ b/.github/PULL_REQUEST_TEMPLATE.md @@ -0,0 +1,6 @@ +Fixes # . + +Changes proposed in this pull request: + - + - + - \ No newline at end of file diff --git a/.github/dependabot.yml b/.github/dependabot.yml new file mode 100644 index 0000000..539359c --- /dev/null +++ b/.github/dependabot.yml @@ -0,0 +1,12 @@ +# To get started with Dependabot version updates, you'll need to specify which +# package ecosystems to update and where the package manifests are located. +# Please see the documentation for all configuration options: +# https://docs.github.com/github/administering-a-repository/configuration-options-for-dependency-updates + +version: 2 +updates: + - package-ecosystem: "npm" # See documentation for possible values + directory: "/" # Location of package manifests + schedule: + interval: "weekly" + diff --git a/.github/workflows/dependabot_automerge.yml b/.github/workflows/dependabot_automerge.yml new file mode 100644 index 0000000..b73ff6a --- /dev/null +++ b/.github/workflows/dependabot_automerge.yml @@ -0,0 +1,14 @@ +name: auto-merge + +on: + pull_request: + +jobs: + auto-merge: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v2 + - uses: ahmadnassri/action-dependabot-auto-merge@v2 + with: + target: minor + github-token: ${{ secrets.mytoken }} \ No newline at end of file diff --git a/.github/workflows/tests.yaml b/.github/workflows/tests.yaml new file mode 100644 index 0000000..1870f3c --- /dev/null +++ b/.github/workflows/tests.yaml @@ -0,0 +1,21 @@ +name: 'Unit Tests' + +on: + workflow_dispatch: + push: + branches: + - main + pull_request: + +jobs: + tests: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v3 + with: + submodules: true + - uses: actions/setup-node@v3 + with: + node-version: current + - run: npm ci + - run: npm test diff --git a/.gitignore b/.gitignore index c6bba59..57d9462 100644 --- a/.gitignore +++ b/.gitignore @@ -89,7 +89,6 @@ out # Nuxt.js build / generate output .nuxt -dist # Gatsby files .cache/ diff --git a/.npmignore b/.npmignore new file mode 100644 index 0000000..338e802 --- /dev/null +++ b/.npmignore @@ -0,0 +1,114 @@ +# Logs +logs +*.log +npm-debug.log* +yarn-debug.log* +yarn-error.log* +lerna-debug.log* + +# Diagnostic reports (https://nodejs.org/api/report.html) +report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json + +# Runtime data +pids +*.pid +*.seed +*.pid.lock + +# Directory for instrumented libs generated by jscoverage/JSCover +lib-cov + +# Coverage directory used by tools like istanbul +coverage +*.lcov + +# nyc test coverage +.nyc_output + +# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files) +.grunt + +# Bower dependency directory (https://bower.io/) +bower_components + +# node-waf configuration +.lock-wscript + +# Compiled binary addons (https://nodejs.org/api/addons.html) +build/Release + +# Dependency directories +node_modules/ +jspm_packages/ + +# TypeScript v1 declaration files +typings/ + +# TypeScript cache +*.tsbuildinfo + +# Optional npm cache directory +.npm + +# Optional eslint cache +.eslintcache + +# Microbundle cache +.rpt2_cache/ +.rts2_cache_cjs/ +.rts2_cache_es/ +.rts2_cache_umd/ + +# Optional REPL history +.node_repl_history + +# Output of 'npm pack' +*.tgz + +# Yarn Integrity file +.yarn-integrity + +# dotenv environment variables file +.env +.env.test + +# parcel-bundler cache (https://parceljs.org/) +.cache + +# Next.js build output +.next + +# Nuxt.js build / generate output +.nuxt + +# Gatsby files +.cache/ +# Comment in the public line in if your project uses Gatsby and *not* Next.js +# https://nextjs.org/blog/next-9-1#public-directory-support +# public + +# vuepress build output +.vuepress/dist + +# Serverless directories +.serverless/ + +# FuseBox cache +.fusebox/ + +# DynamoDB Local files +.dynamodb/ + +# TernJS port file +.tern-port +deps +samples/output*.md +samples/raw*.txt +samples/*.sql +samples/*/*.sql +junit.xml +*.bak +output-*.sql +tests +build +funktechno-little-mermaid-2-the-sql* \ No newline at end of file diff --git a/.vscode/launch.json b/.vscode/launch.json new file mode 100644 index 0000000..73cac5d --- /dev/null +++ b/.vscode/launch.json @@ -0,0 +1,53 @@ +{ + // Use IntelliSense to learn about possible attributes. + // Hover to view descriptions of existing attributes. + // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 + "version": "0.2.0", + "configurations": [ + { + "type": "node", + "request": "launch", + "name": "Debug cli", + "program": "${workspaceFolder}\\src\\index.ts", + "preLaunchTask": "npm: build", + "sourceMaps": true, + "smartStep": true, + "internalConsoleOptions": "openOnSessionStart", + "outFiles": [ + "${workspaceFolder}/lib/**/*.js" + ], + "args": ["--database","sqlite", "--src", "samples/chinook-database-2.0.1.md"] + }, + { + "type": "node", + "request": "launch", + "name": "Jest single run", + "program": "${workspaceRoot}/node_modules/jest/bin/jest.js", + "args": [ + "-c", + "./jest.config.ts", + "--verbose", + "-i", + "--no-cache" + ], + "console": "integratedTerminal", + "internalConsoleOptions": "neverOpen" + }, + { + "type": "node", + "request": "launch", + "name": "Jest watch run", + "program": "${workspaceRoot}/node_modules/jest/bin/jest.js", + "args": [ + "-c", + "./jest.config.ts", + "--verbose", + "-i", + "--no-cache", + "--watchAll" + ], + "console": "integratedTerminal", + "internalConsoleOptions": "neverOpen" + } + ] +} \ No newline at end of file diff --git a/README.md b/README.md index becd207..4544b97 100644 --- a/README.md +++ b/README.md @@ -1,2 +1,5 @@ # sqltooling-drawio plugins for sql tooling in drawio + +## Getting Started +* see https://github.com/ariel-bentu/tam-drawio for install options with vscode \ No newline at end of file diff --git a/build/createDistFolder.js b/build/createDistFolder.js new file mode 100644 index 0000000..ea36a20 --- /dev/null +++ b/build/createDistFolder.js @@ -0,0 +1,16 @@ +const fs = require('fs'); +const path = require('path'); + +const dir = 'dist'; +// path.resolve(path.join(__dirname, 'dist')); + +if (!fs.existsSync(dir)) { + fs.mkdirSync(dir); +} + +// OR +if (!fs.existsSync(dir)) { + fs.mkdirSync(dir, { + mode: 0o744, // Not supported on Windows. Default: 0o777 + }); +} \ No newline at end of file diff --git a/dist/sql.js b/dist/sql.js new file mode 100644 index 0000000..9f8aebb --- /dev/null +++ b/dist/sql.js @@ -0,0 +1,1459 @@ +/** + * Parse SQL CREATE TABLE. Simple initial version for community to improve. + */ +Draw.loadPlugin(function(ui) { + + // export sql methods + + /** + * Mermaid Models TO SQL parser + * src https://github.com/Software-Developers-IRL/Little-Mermaid-2-The-SQL/blob/main/src/generate-sql-ddl.ts + */ + class DbParser { + constructor(dbType, db) { + this.db = db; + this.dbType = dbType; + } + /** + * return sql from mermaid erDiagram db models + * @returns + */ + getSQLDataDefinition() { + this.entities = this.db.getEntities(); + this.relationships = this.db.getRelationships(); + return this.lexer(); + } + lexer() { + let statementGeneration = []; + if (this.entities) { + for (const key in this.entities) { + if (Object.prototype.hasOwnProperty.call(this.entities, key)) { + const entity = this.entities[key]; + statementGeneration.push(this.createTable(key, entity)); + } + } + } + return statementGeneration.join(""); + } + /** + * convert labels with start and end strings per database type + * @param label + * @returns + */ + dbTypeEnds(label) { + let char1 = '"'; + let char2 = '"'; + if (this.dbType == "mysql") { + char1 = "`"; + char2 = "`"; + } + else if (this.dbType == "sqlserver") { + char1 = "["; + char2 = "]"; + } + return `${char1}${label}${char2}`; + } + /** + * generate create table statement + * also includes primary keys and foreign keys + * @param entityKey + * @param entity + * @returns + */ + createTable(entityKey, entity) { + let statement = `CREATE TABLE ${this.dbTypeEnds(entityKey)} (`; + let primaryKeys = []; + let attributesAdded = 0; + for (let i = 0; i < entity.attributes.length; i++) { + const attribute = entity.attributes[i]; + if (attribute.attributeType && attribute.attributeName) { + statement += attributesAdded == 0 ? "\n" : ",\n"; + attributesAdded++; + // need to add parenthesis or commas + // BIGGEST difference from original is that column types don't have to be fixed + let columnType = attribute.attributeType.trim(); + if (attribute.attributeComment) { + let attributeName = attribute.attributeName; + let attributeComment = attribute.attributeComment.trim(); + if (attribute.attributeComment.indexOf("'") != -1) { + // extract + const testFullNameMatches = attribute.attributeComment.match(/(?<=((?<=[\s,.:;"']|^)["']))(?:(?=(\\?))\2.)*?(?=\1)/gmu); + if (testFullNameMatches && testFullNameMatches.length > 0) { + attributeName = testFullNameMatches[0]; + attributeComment = attributeComment + .replace(`'${attributeName}'`, "") + .trim(); + } + } + if (attributeComment) + attributeComment = " " + attributeComment; + // check if contains full column name + statement += `\t${this.dbTypeEnds(attributeName)} ${columnType}${attributeComment}`; + if (attribute.attributeKeyType && + attribute.attributeKeyType == "PK") { + primaryKeys.push(attribute.attributeName); + } + } + else { + if (attribute.attributeKeyType && + attribute.attributeKeyType == "PK") { + primaryKeys.push(attribute.attributeName); + } + statement += `\t${this.dbTypeEnds(attribute.attributeName)} ${columnType}`; + } + } + } + if (primaryKeys.length > 0) { + statement += ",\n\tPRIMARY KEY("; + for (let i = 0; i < primaryKeys.length; i++) { + const element = primaryKeys[i]; + statement += (i == 0 ? "" : ",") + this.dbTypeEnds(primaryKeys[i]); + } + statement += ")"; + } + // foreign keys + let entityFKeys = this.relationships.filter((relation) => relation.entityB == entityKey); + if (entityFKeys.length > 0) { + for (let i = 0; i < entityFKeys.length; i++) { + const fk = entityFKeys[i]; + const fkRelTxt = fk.roleA; + // must match format "[..] to [..]" + const keySplit = "] to ["; + if (fkRelTxt.indexOf(keySplit) != -1 && + fkRelTxt[0] == "[" && + fkRelTxt[fkRelTxt.length - 1] == "]") { + let keys = fkRelTxt.substring(1, fkRelTxt.length - 1).split(keySplit); + // remove quotes + let fkCol = keys[1].replace(/[\'\"]/gim, ""); + if (fkCol.indexOf(".") != -1) { + fkCol = fkCol.split(".")[1]; + } + fkCol = fkCol.trim(); + let pkCol = keys[0].replace(/[\'\"]/gim, ""); + if (pkCol.indexOf(".") != -1) { + pkCol = pkCol.split(".")[1]; + } + pkCol = pkCol.trim(); + statement += `,\n\tFOREIGN KEY (${this.dbTypeEnds(fkCol)}) REFERENCES ${this.dbTypeEnds(fk.entityA)}(${this.dbTypeEnds(pkCol)})`; + } + } + } + if (attributesAdded != 0) { + statement += "\n"; + } + statement += `);\n\n`; + return statement; + } + } + + //Create Base div + const divGenSQL = document.createElement('div'); + divGenSQL.style.userSelect = 'none'; + divGenSQL.style.overflow = 'hidden'; + divGenSQL.style.padding = '10px'; + divGenSQL.style.height = '100%'; + + const sqlInputGenSQL = document.createElement('textarea'); + sqlInputGenSQL.style.height = '200px'; + sqlInputGenSQL.style.width = '100%'; + sqlInputGenSQL.value = '-- click a database type button' + mxUtils.br(divGenSQL); + divGenSQL.appendChild(sqlInputGenSQL); + var theMenuExportAs = ui.menus.get('exportAs'); + let buttonLabel = 'tosql=To SQL' + // vscode extension support + if(!(theMenuExportAs && theMenuExportAs.enabled)) { + buttonLabel = 'tosql=Export As SQL' + } + // Extends Extras menu + mxResources.parse(buttonLabel); + + const wndGenSQL = new mxWindow(mxResources.get('tosql'), divGenSQL, document.body.offsetWidth - 480, 140, + 320, 320, true, true); + wndGenSQL.destroyOnClose = false; + wndGenSQL.setMaximizable(false); + wndGenSQL.setResizable(false); + wndGenSQL.setClosable(true); + + /** + * return text quantifiers for dialect + * @returns json + */ + function GetColumnQuantifiers(type) { + let chars = { + Start: '"', + End: '"', + }; + if (type == "mysql") { + chars.Start = "`"; + chars.End = "`"; + } + else if (type == "sqlserver") { + chars.Start = "["; + chars.End = "]"; + } + return chars; + } + /** + * sometimes rows have spans or styles, an attempt to remove them + * @param {*} label + * @returns + */ + function removeHtml(label){ + var div = document.createElement("div"); + divGenSQL.innerHTML = label; + var text = div.textContent || div.innerText || ""; + return text; + } + /** + * extract row column attributes + * @param {*} label + * @param {*} columnQuantifiers + * @returns + */ + function getDbLabel(label, columnQuantifiers){ + label = removeHtml(label) + // fix duplicate spaces and different space chars + label = label + .replace(/\s+/g, " ") + let firstSpaceIndex = label[0] == columnQuantifiers.Start && + label.indexOf(columnQuantifiers.End + " ") !== -1 + ? label.indexOf(columnQuantifiers.End + " ") + : label.indexOf(" "); + let attributeType = label.substring(firstSpaceIndex + 1).trim(); + let attributeName = RemoveNameQuantifiers(label.substring(0, firstSpaceIndex)); + let attribute = { + attributeName, + attributeType + } + return attribute + } + function RemoveNameQuantifiers(name) { + return name.replace(/\[|\]|\(|\"|\'|\`/g, "").trim(); + } + + function getMermaidDiagramDb(type){ + var model = ui.editor.graph.getModel() + // same models from mermaid for diagram relationships + // only difference is entities is an array rather than object to allow duplicate tables + let entities = {} + let relationships = [] + // build models + for (const key in model.cells) { + if (Object.hasOwnProperty.call(model.cells, key)) { + const mxcell = model.cells[key]; + if(mxcell.mxObjectId.indexOf("mxCell") !== -1) { + if(mxcell.style && mxcell.style.trim().startsWith("swimlane;")){ + let entity = { + name: RemoveNameQuantifiers(mxcell.value), + attributes: [] + } + for (let c = 0; c < mxcell.children.length; c++) { + const col = mxcell.children[c]; + if(col.mxObjectId.indexOf("mxCell") !== -1) { + if(col.style && col.style.trim().startsWith("shape=partialRectangle")){ + const columnQuantifiers = GetColumnQuantifiers(type); + //Get delimiter of column name + //Get full name + let attribute = getDbLabel(col.value, columnQuantifiers) + var attributeKeyType = col.children.find(x=> ["FK","PK"].findIndex(k => k== x.value.toUpperCase()) !== -1 || + x.value.toUpperCase().indexOf("PK,")!=-1) + if(attributeKeyType){ + attribute.attributeKeyType = attributeKeyType.value + if(attribute.attributeKeyType != "PK" && attribute.attributeKeyType.indexOf("PK") != -1){ + attribute.attributeKeyType = "PK" + } + } + entity.attributes.push(attribute) + if(col.edges && col.edges.length){ + // check for edges foreign keys + for (let e = 0; e < col.edges.length; e++) { + const edge = col.edges[e]; + if(edge.mxObjectId.indexOf("mxCell") !== -1) { + if(edge.style && edge.style.indexOf("endArrow=") != -1 && edge.source && + edge.source.value && edge.target && edge.target.value){ + // need to check if end is open or certain value to determin relationship type + // extract endArrow txt + // check if both match and contain many or open + // if both match and are many then create a new table + let endCheck = "endArrow=" + let endArr = edge.style.indexOf(endCheck) != -1 ? + edge.style.substring(edge.style.indexOf(endCheck) + endCheck.length, edge.style.substring(edge.style.indexOf(endCheck) + endCheck.length).indexOf(";") + edge.style.indexOf(endCheck) + endCheck.length) + : "" + let startCheck = "startArrow=" + let startArr = edge.style.indexOf(startCheck) != -1 ? + edge.style.substring(edge.style.indexOf(startCheck) + startCheck.length, edge.style.substring(edge.style.indexOf(startCheck) + startCheck.length).indexOf(";") + edge.style.indexOf(startCheck) + startCheck.length) + : "" + + var manyCheck = ["open","many"] + var sourceIsPrimary = endArr && manyCheck + .findIndex(x => endArr.toLocaleLowerCase().indexOf(x)!=-1) != -1; + var targetIsPrimary = startArr && manyCheck + .findIndex(x => startArr.toLocaleLowerCase().indexOf(x)!=-1) != -1; + // has to be one to many and not one to one + if((targetIsPrimary || sourceIsPrimary) && + !(targetIsPrimary && sourceIsPrimary) + ){ + var sourceId = edge.source.value; + sourceAttr = getDbLabel(sourceId, columnQuantifiers); + sourceId = sourceAttr.attributeName + var sourceEntity = RemoveNameQuantifiers(edge.source.parent.value); + var targetId = edge.target.value; + targetAttr = getDbLabel(targetId, columnQuantifiers); + targetId = targetAttr.attributeName + var targetEntity = RemoveNameQuantifiers(edge.target.parent.value); + // entityA primary + // entityB foreign + let relationship = { + entityA: sourceIsPrimary ? sourceEntity : targetEntity, + entityB: sourceIsPrimary ? targetEntity : sourceEntity, + // based off of styles? + relSpec: { + cardA: 'ZERO_OR_MORE', + cardB: 'ONLY_ONE', + relType: "IDENTIFYING" + }, + roleA: sourceIsPrimary ? + `[${sourceEntity}.${sourceId}] to [${targetEntity}.${targetId}]` : + `[${targetEntity}.${targetId}] to [${sourceEntity}.${sourceId}]` + } + // check that is doesn't already exist + var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA) + if(exists ==-1){ + relationships.push(relationship) + } + } else if(targetIsPrimary && sourceIsPrimary){ + // add a new many to many table + var sourceId = edge.source.value; + sourceAttr = getDbLabel(sourceId, columnQuantifiers); + sourceAttr.attributeKeyType = "PK" + sourceId = sourceAttr.attributeName + var sourceEntity = RemoveNameQuantifiers(edge.source.parent.value); + var targetId = edge.target.value; + targetAttr = getDbLabel(targetId, columnQuantifiers); + targetAttr.attributeKeyType = "PK" + targetId = targetAttr.attributeName + var targetEntity = RemoveNameQuantifiers(edge.target.parent.value); + let compositeEntity = { + name: RemoveNameQuantifiers(sourceEntity) + "_" + RemoveNameQuantifiers(targetEntity), + attributes: [sourceAttr, targetAttr] + } + // add composite entity + if(entities[compositeEntity.name]){ + // DON'T add duplicate composite tables + } else { + entities[compositeEntity.name] = compositeEntity + } + // entityA primary + // entityB foreign + let relationship = { + entityA: sourceEntity, + entityB: compositeEntity.name, + // based off of styles? + relSpec: { + cardA: 'ZERO_OR_MORE', + cardB: 'ONLY_ONE', + relType: "IDENTIFYING" + }, + roleA: `[${sourceEntity}.${sourceId}] to [${compositeEntity.name}.${sourceId}]` + } + // check that is doesn't already exist + var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA) + if(exists ==-1){ + relationships.push(relationship) + } + let relationship2 = { + entityA: targetEntity, + entityB: compositeEntity.name, + // based off of styles? + relSpec: { + cardA: 'ZERO_OR_MORE', + cardB: 'ONLY_ONE', + relType: "IDENTIFYING" + }, + roleA: `[${targetEntity}.${targetId}] to [${compositeEntity.name}.${targetId}]` + } + // check that is doesn't already exist + exists = relationships.findIndex(r => r.entityA == relationship2.entityA && r.entityB == relationship2.entityB && r.roleA == relationship2.roleA) + if(exists ==-1){ + relationships.push(relationship2) + } + } + + } + } + + } + } + } + } + } + // allows for duplicates if another table has the same name + if(entities[entity.name]){ + var count = 2; + while(entities[entity.name + count.toString()]){ + count++; + } + entities[entity.name + count.toString()] = entity + } else { + entities[entity.name] = entity + } + } + + } + } + } + + class DbDefinition{ + constructor(entities, relationships){ + this.entities = entities; + this.relationships = relationships; + } + + getEntities(){ + return this.entities + } + + getRelationships(){ + return this.relationships + } + } + + var db = new DbDefinition(entities, relationships); + + return db; + } + + function generateSql(type) { + + // get diagram model + var db = getMermaidDiagramDb(type); + // load parser + var parser = new DbParser(type, db) + // generate sql + var sql = parser.getSQLDataDefinition() + sql = `/*\n\tGenerated in drawio\n\tDatabase: ${type}\n*/\n\n` + sql + sql = sql.trim(); + // update sql value in text area + sqlInputGenSQL.value = sql; + // TODO: use selection as well? + var modelSelected = ui.editor.graph.getSelectionModel() + }; + + mxUtils.br(divGenSQL); + + const resetBtnGenSQL = mxUtils.button(mxResources.get('reset'), function() { + sqlInputGenSQL.value = ''; + }); + + resetBtnGenSQL.style.marginTop = '8px'; + resetBtnGenSQL.style.marginRight = '4px'; + resetBtnGenSQL.style.padding = '4px'; + divGenSQL.appendChild(resetBtnGenSQL); + + const btnGenSQL_mysql = mxUtils.button('MySQL', function() { + generateSql('mysql'); + }); + + btnGenSQL_mysql.style.marginTop = '8px'; + btnGenSQL_mysql.style.padding = '4px'; + divGenSQL.appendChild(btnGenSQL_mysql); + + const btnGenSQL_sqlserver = mxUtils.button('SQL Server', function() { + generateSql('sqlserver'); + }); + + btnGenSQL_sqlserver.style.marginTop = '8px'; + btnGenSQL_sqlserver.style.padding = '4px'; + divGenSQL.appendChild(btnGenSQL_sqlserver); + + const btnGenSQL_postgres = mxUtils.button('PostgreSQL', function() { + generateSql('postgres'); + }); + + btnGenSQL_postgres.style.marginTop = '8px'; + btnGenSQL_postgres.style.padding = '4px'; + divGenSQL.appendChild(btnGenSQL_postgres); + + const btnGenSQL_sqlite = mxUtils.button('Sqlite', function() { + generateSql('sqlite'); + }); + + btnGenSQL_sqlite.style.marginTop = '8px'; + btnGenSQL_sqlite.style.padding = '4px'; + divGenSQL.appendChild(btnGenSQL_sqlite); + + // Adds action + ui.actions.addAction('tosql', function() { + wndGenSQL.setVisible(!wndGenSQL.isVisible()); + + if (wndGenSQL.isVisible()) { + sqlInputGenSQL.focus(); + } + }); + // end export sql methods + + // import diagrams from sql text methods + + const contants_1 ={} + contants_1.CONSTRAINT_Foreign_Key = contants_1.CONSTRAINT_Primary_Key = contants_1.Foreign_Key = contants_1.Primary_Key = contants_1.CONSTRAINT = contants_1.AlterTable = contants_1.CreateTable = void 0; + contants_1.CreateTable = "create table"; + contants_1.AlterTable = "alter table "; + contants_1.CONSTRAINT = "constraint"; + contants_1.Primary_Key = "primary key"; + contants_1.Foreign_Key = "foreign key"; + contants_1.CONSTRAINT_Primary_Key = "constraint primary key"; + contants_1.CONSTRAINT_Foreign_Key = "constraint foreign key"; + + /** + * Main Parser class + */ + class SqlSimpleParser { + /** + * Parser constructor. + * Default dialect is 'sqlite'. + * Options: "mysql" | "sqlite" | "postgres" | "sqlserver" . + * + * @param dialect SQL dialect ('sqlite'). + */ + constructor(dialect = "sqlite") { + this.tableList = []; + /** + * Parsed statements. + */ + this.statements = []; + /** + * Remains of string feed, after last parsed statement. + */ + this.remains = ""; + /** + * Whether preparser is currently escaped. + */ + this.escaped = false; + /** + * Current quote char of preparser. + */ + this.quoted = ""; + this.exportedTables = 0; + this.SQLServer = "sqlserver"; + this.MODE_SQLSERVER = false; + this.foreignKeyList = []; + this.primaryKeyList = []; + this.dialect = dialect; + this.MODE_SQLSERVER = + this.dialect !== undefined && + this.dialect !== null && + this.dialect == this.SQLServer; + } + /** + * Feed chunk of string into parser. + * + * @param chunk Chunk of string to be parsed. + */ + feed(chunk) { + // + const removedComments = chunk + // remove database comments, multiline, --, and // + .replace(/\/\*[\s\S]*?\*\/|\/\/|--.*/g, "") + .replace(/IF NOT EXISTS/gi, "") + .trim(); + const cleanedLines = removedComments + .split("\n") + // remove empty lines + .filter((n) => n) + // remove multiple spaces + .map((n) => n.replace(/\s+/g, " ").trim()); + // combine lines that are in parenthesis + const lines = []; + let insertSameLine = false; + cleanedLines.forEach((n) => { + if (lines.length > 0){ + if((n[0] == "(" && + lines[lines.length - 1].toLocaleLowerCase().indexOf(contants_1.CreateTable) == + -1) || + insertSameLine) { + if (lines.length > 0) { + insertSameLine = true; + lines[lines.length - 1] += ` ${n}`; + if (n[0] == ")") + insertSameLine = false; + } + } + else if(lines[lines.length - 1].match(/CONSTRAINT/gi) && + (n.match(/FOREIGN KEY/gi) && !n.match(/CONSTRAINT/gi)) + ){ + lines[lines.length - 1] += ` ${n}`; + } + // add to previous line if current has references and previous has foreign key + else if(lines[lines.length - 1].match(/FOREIGN KEY/gi) && + (n.match(/REFERENCES/gi) && !n.match(/FOREIGN KEY/gi)) + ){ + lines[lines.length - 1] += ` ${n}`; + } + else if(n.substring(0,2).toUpperCase() == "ON"){ + lines[lines.length - 1] += ` ${n}`; + } + else { + lines.push(n); + } + } + else { + lines.push(n); + } + }); + let currentTableModel = null; + //Parse SQL to objects + for (let i = 0; i < lines.length; i++) { + // rowCell = null; + const tmp = lines[i].trim(); + const propertyRow = tmp.toLowerCase().trim(); + if (propertyRow[0] == ")") { + // close table + if (currentTableModel) { + this.tableList.push(currentTableModel); + currentTableModel = null; + } + continue; + } + //Parse Table + if (propertyRow.indexOf(contants_1.CreateTable) != -1) { + //Parse row + let name = tmp + .replace(this.stringToRegex(`/${contants_1.CreateTable}/gi`), "") + .trim(); + //Parse Table Name + name = this.ParseTableName(name); + if (currentTableModel !== null) { + //Add table to the list + this.tableList.push(currentTableModel); + } + //Create Table + currentTableModel = this.CreateTable(name); + } + // Parse Properties + else if (tmp !== "(" && + currentTableModel != null && + propertyRow.indexOf(contants_1.AlterTable) == -1) { + //Parse the row + let name = tmp.substring(0, tmp.charAt(tmp.length - 1) === "," ? tmp.length - 1 : tmp.length); + //Attempt to get the Key Type + let propertyType = name.toLowerCase(); + // .substring(0, AlterTable.length).toLowerCase(); + //Add special constraints + if (this.MODE_SQLSERVER) { + if (propertyType.indexOf(contants_1.CONSTRAINT) !== -1 && + propertyType.indexOf(contants_1.Primary_Key) !== -1) { + propertyType = contants_1.CONSTRAINT_Primary_Key; + } + if (propertyType.indexOf(contants_1.CONSTRAINT) !== -1 && + propertyType.indexOf(contants_1.Foreign_Key) !== -1) { + propertyType = contants_1.CONSTRAINT_Foreign_Key; + } + } + //Verify if this is a property that doesn't have a relationship (One minute of silence for the property) + const normalProperty = !propertyType.match(/PRIMARY KEY\s?\(/gi) && + propertyType.indexOf(contants_1.Foreign_Key) == -1 && + propertyType.indexOf(contants_1.CONSTRAINT_Primary_Key) == -1 && + propertyType.indexOf(contants_1.CONSTRAINT_Foreign_Key) == -1; + const nameSkipCheck = name.toUpperCase().trim(); + //Parse properties that don't have relationships + if (normalProperty) { + if (name === "" || name === "" || name === ");") { + continue; + } + let ExtendedProperties = null; + if (this.MODE_SQLSERVER) { + if (nameSkipCheck.indexOf(" ASC") !== -1 || + nameSkipCheck.indexOf(" DESC") !== -1 || + nameSkipCheck.indexOf("EXEC ") !== -1 || + nameSkipCheck.indexOf("WITH ") !== -1 || + nameSkipCheck.indexOf(" ON") !== -1 || + nameSkipCheck.indexOf("ALTER ") !== -1 || + // comments already removed + nameSkipCheck.indexOf("/*") !== -1 || + nameSkipCheck.indexOf(" CONSTRAINT") !== -1 || + nameSkipCheck.indexOf("SET ") !== -1 || + nameSkipCheck.indexOf(" NONCLUSTERED") !== -1 || + // no spaces desired + nameSkipCheck.indexOf("GO") !== -1 || + nameSkipCheck.indexOf("REFERENCES ") !== -1) { + continue; + } + //Get delimiter of column name + const firstSpaceIndex = name[0] == "[" && name.indexOf("]" + " ") !== -1 + ? name.indexOf("]" + " ") + : name.indexOf(" "); + ExtendedProperties = name.substring(firstSpaceIndex + 1).trim(); + //Get full name + name = name.substring(0, firstSpaceIndex); + name = this.RemoveNameQuantifiers(name); + } + else { + const columnQuantifiers = this.GetColumnQuantifiers(); + //Get delimiter of column name + const firstSpaceIndex = name[0] == columnQuantifiers.Start && + name.indexOf(columnQuantifiers.End + " ") !== -1 + ? name.indexOf(columnQuantifiers.End + " ") + : name.indexOf(" "); + ExtendedProperties = name.substring(firstSpaceIndex + 1).trim(); + //Get full name + name = name.substring(0, firstSpaceIndex); + name = this.RemoveNameQuantifiers(name); + } + //Create Property + const propertyModel = this.CreateProperty(name, currentTableModel.Name, null, false, ExtendedProperties); + //Add Property to table + currentTableModel.Properties.push(propertyModel); + if (ExtendedProperties.toLocaleLowerCase().indexOf(contants_1.Primary_Key) > -1) { + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(name, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + } + } + else { + //Parse Primary Key + if (propertyType.indexOf(contants_1.Primary_Key) != -1 || + propertyType.indexOf(contants_1.CONSTRAINT_Primary_Key) != -1) { + if (!this.MODE_SQLSERVER) { + const primaryKey = name + .replace(/PRIMARY KEY\s?\(/gi, "") + .replace(")", ""); + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(primaryKey, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + } + else { + if (propertyRow.indexOf(contants_1.Primary_Key) !== -1 && + nameSkipCheck.indexOf("CLUSTERED") === -1) { + const primaryKey = name + .replace(/PRIMARY KEY\s?\(/gi, "") + .replace(")", ""); + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(primaryKey, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + } + else { + const startIndex = name.toLocaleLowerCase().indexOf("("); + const endIndex = name.indexOf(")") + 1; + const primaryKey = name + .substring(startIndex, endIndex) + .replace("(", "") + .replace(")", "") + .replace(/ASC/gi, "") + .trim(); + const columnQuantifiers = this.GetColumnQuantifiers(); + //Get delimiter of column name + const firstSpaceIndex = primaryKey[0] == columnQuantifiers.Start && + primaryKey.indexOf(columnQuantifiers.End + " ") !== -1 + ? primaryKey.indexOf(columnQuantifiers.End + " ") + : primaryKey.indexOf(" "); + const primaryKeyRow = firstSpaceIndex == -1 + ? primaryKey + : primaryKey.substring(firstSpaceIndex + 1).trim(); + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(primaryKeyRow, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + } + } + } + } + //Parse Foreign Key + if (propertyType.indexOf(contants_1.Foreign_Key) != -1 || + propertyType.indexOf(contants_1.CONSTRAINT_Foreign_Key) != -1) { + if (!this.MODE_SQLSERVER || propertyRow.indexOf(contants_1.AlterTable) == -1) { + this.ParseMySQLForeignKey(name, currentTableModel); + } + else { + let completeRow = name; + if (nameSkipCheck.indexOf("REFERENCES") === -1) { + const referencesRow = lines[i + 1].trim(); + completeRow = + "ALTER TABLE [dbo].[" + + currentTableModel.Name + + "] WITH CHECK ADD" + + " " + + name + + " " + + referencesRow; + } + this.ParseSQLServerForeignKey(completeRow, currentTableModel); + } + } + } + else if (propertyRow.indexOf(contants_1.AlterTable) != -1) { + if (this.MODE_SQLSERVER) { + //Parse the row + const alterTableRow = tmp.substring(0, tmp.charAt(tmp.length - 1) === "," ? tmp.length - 1 : tmp.length); + const referencesRow = lines[i + 1].trim(); + const completeRow = alterTableRow + " " + referencesRow; + this.ParseSQLServerForeignKey(completeRow, currentTableModel); + } + } + } + // parse fk and primary keys + if (this.primaryKeyList.length > 0) { + this.primaryKeyList.forEach((pk) => { + // find table index + const pkTableIndex = this.tableList.findIndex((t) => t.Name.toLocaleLowerCase() == + pk.PrimaryKeyTableName.toLocaleLowerCase()); + // find property index + if (pkTableIndex > -1) { + const propertyIndex = this.tableList[pkTableIndex].Properties.findIndex((p) => p.Name.toLocaleLowerCase() == + pk.PrimaryKeyName.toLocaleLowerCase()); + if (propertyIndex > -1) { + this.tableList[pkTableIndex].Properties[propertyIndex].IsPrimaryKey = true; + } + } + }); + } + if (this.foreignKeyList.length > 0) { + this.foreignKeyList.forEach((fk) => { + // find table index + const pkTableIndex = this.tableList.findIndex((t) => t.Name.toLocaleLowerCase() == + fk.ReferencesTableName.toLocaleLowerCase()); + // find property index + if (pkTableIndex > -1) { + const propertyIndex = this.tableList[pkTableIndex].Properties.findIndex((p) => p.Name.toLocaleLowerCase() == + fk.PrimaryKeyName.toLocaleLowerCase()); + if (propertyIndex > -1) { + this.tableList[pkTableIndex].Properties[propertyIndex].ForeignKey.push(fk); + if (!fk.IsDestination) { + this.tableList[pkTableIndex].Properties[propertyIndex].IsForeignKey = true; + } + } + } + }); + } + return this; + } + stringToRegex(str) { + // Main regex + const mainResult = str.match(/\/(.+)\/.*/); + const optionsResult = str.match(/\/.+\/(.*)/); + if (mainResult && optionsResult) { + const main = mainResult[1]; + // Regex options + const options = optionsResult[1]; + // Compiled regex + return new RegExp(main, options); + } + return new RegExp("//(.+)/.*/", "//.+/(.*)/"); + } + CreatePrimaryKey(primaryKeyName, primaryKeyTableName) { + const primaryKeyNames = this.RemoveNameQuantifiers(primaryKeyName) + .split(",") + .filter((n) => n) + // remove multiple spaces + .map((n) => n.replace(/\s+/g, " ").trim()); + const primaryKeys = []; + primaryKeyNames.forEach(name => { + const primaryKey = { + PrimaryKeyTableName: primaryKeyTableName, + PrimaryKeyName: name, + }; + primaryKeys.push(primaryKey); + }); + return primaryKeys; + } + CreateProperty(name, tableName, foreignKey, isPrimaryKey, columnProps) { + const isForeignKey = foreignKey !== undefined && foreignKey !== null; + const property = { + Name: name, + ColumnProperties: columnProps, + TableName: tableName, + ForeignKey: foreignKey || [], + IsForeignKey: isForeignKey, + IsPrimaryKey: isPrimaryKey, + }; + return property; + } + ParseMySQLForeignKey(name, currentTableModel) { + const referencesIndex = name.toLowerCase().indexOf("references"); + let foreignKeySQL = name.substring(0, referencesIndex); + foreignKeySQL = foreignKeySQL.substring(foreignKeySQL.toUpperCase().indexOf("FOREIGN KEY")) + let referencesSQL = name.substring(referencesIndex, name.length); + //Remove references syntax + referencesSQL = referencesSQL.replace(/REFERENCES /gi, ""); + //Get Table and Property Index + const referencedTableIndex = referencesSQL.indexOf("("); + const referencedPropertyIndex = referencesSQL.indexOf(")"); + //Get Referenced Table + const referencedTableName = referencesSQL.substring(0, referencedTableIndex); + //Get Referenced Key + const referencedPropertyName = referencesSQL.substring(referencedTableIndex + 1, referencedPropertyIndex); + //Get ForeignKey + const foreignKey = foreignKeySQL + .replace(/FOREIGN KEY\s?\(/gi, "") + .replace(")", "") + .replace(" ", ""); + //Create ForeignKey + const foreignKeyOriginModel = this.CreateForeignKey(foreignKey, currentTableModel.Name, referencedPropertyName, referencedTableName, true); + //Add ForeignKey Origin + this.foreignKeyList.push(foreignKeyOriginModel); + //Create ForeignKey + const foreignKeyDestinationModel = this.CreateForeignKey(referencedPropertyName, referencedTableName, foreignKey, currentTableModel.Name, false); + //Add ForeignKey Destination + this.foreignKeyList.push(foreignKeyDestinationModel); + } + ParseSQLServerForeignKey(name, currentTableModel) { + const referencesIndex = name.toLowerCase().indexOf("references"); + let foreignKeySQL = ""; + if (name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`) !== -1) { + foreignKeySQL = name + .substring(name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`), referencesIndex) + .replace(/FOREIGN KEY\(/gi, "") + .replace(")", ""); + } + else { + foreignKeySQL = name + .substring(name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`), referencesIndex) + .replace(/FOREIGN KEY\s?\(/gi, "") + .replace(")", ""); + } + let referencesSQL = name.substring(referencesIndex, name.length); + const nameSkipCheck = name.toUpperCase().trim(); + let alterTableName = name + .substring(0, nameSkipCheck.indexOf("WITH")) + .replace(/ALTER TABLE /gi, ""); + if (referencesIndex !== -1 && + alterTableName !== "" && + foreignKeySQL !== "" && + referencesSQL !== "") { + //Remove references syntax + referencesSQL = referencesSQL.replace(/REFERENCES /gi, ""); + //Get Table and Property Index + const referencedTableIndex = referencesSQL.indexOf("("); + const referencedPropertyIndex = referencesSQL.indexOf(")"); + //Get Referenced Table + let referencedTableName = referencesSQL.substring(0, referencedTableIndex); + //Parse Name + referencedTableName = this.ParseSQLServerName(referencedTableName); + //Get Referenced Key + let referencedPropertyName = referencesSQL.substring(referencedTableIndex + 1, referencedPropertyIndex); + //Parse Name + referencedPropertyName = this.ParseSQLServerName(referencedPropertyName); + //Get ForeignKey + let foreignKey = foreignKeySQL + .replace(/FOREIGN KEY\s?\(/gi, "") + .replace(")", ""); + //Parse Name + foreignKey = this.ParseSQLServerName(foreignKey); + //Parse Name + alterTableName = this.ParseSQLServerName(alterTableName); + //Create ForeignKey + const foreignKeyOriginModel = this.CreateForeignKey(foreignKey, alterTableName, referencedPropertyName, referencedTableName, true); + //Add ForeignKey Origin + this.foreignKeyList.push(foreignKeyOriginModel); + //Create ForeignKey + const foreignKeyDestinationModel = this.CreateForeignKey(referencedPropertyName, referencedTableName, foreignKey, alterTableName, false); + //Add ForeignKey Destination + this.foreignKeyList.push(foreignKeyDestinationModel); + } + } + CreateForeignKey(primaryKeyName, primaryKeyTableName, referencesPropertyName, referencesTableName, isDestination) { + const foreignKey = { + PrimaryKeyTableName: this.RemoveNameQuantifiers(primaryKeyTableName), + PrimaryKeyName: this.RemoveNameQuantifiers(primaryKeyName), + ReferencesPropertyName: this.RemoveNameQuantifiers(referencesPropertyName), + ReferencesTableName: this.RemoveNameQuantifiers(referencesTableName), + IsDestination: isDestination !== undefined && isDestination !== null + ? isDestination + : false, + }; + return foreignKey; + } + RemoveNameQuantifiers(name) { + return name.replace(/\[|\]|\(|\"|\'|\`/g, "").trim(); + } + ParseTableName(name) { + if (name.charAt(name.length - 1) === "(") { + name = this.RemoveNameQuantifiers(name); + } + return name; + } + ParseSQLServerName(name, property) { + name = name.replace("[dbo].[", ""); + name = name.replace("](", ""); + name = name.replace("].[", "."); + name = name.replace("[", ""); + if (property == undefined || property == null) { + name = name.replace(" [", ""); + name = name.replace("] ", ""); + } + else { + if (name.indexOf("]") !== -1) { + name = name.substring(0, name.indexOf("]")); + } + } + if (name.lastIndexOf("]") === name.length - 1) { + name = name.substring(0, name.length - 1); + } + if (name.lastIndexOf(")") === name.length - 1) { + name = name.substring(0, name.length - 1); + } + if (name.lastIndexOf("(") === name.length - 1) { + name = name.substring(0, name.length - 1); + } + name = name.replace(" ", ""); + return name; + } + CreateTable(name) { + const table = { + Name: name, + Properties: [], + }; + //Count exported tables + this.exportedTables++; + return table; + } + /** + * Checks whether character is a quotation character. + * + * @param char Character to be evaluated. + */ + static isQuoteChar(char) { + return char === "\"" || char === "'" || char === "`"; + } + /** + * convert labels with start and end strings per database type + * @param label + * @returns + */ + dbTypeEnds(label) { + let char1 = "\""; + let char2 = "\""; + if (this.dialect == "mysql") { + char1 = "`"; + char2 = "`"; + } + else if (this.dialect == "sqlserver") { + char1 = "["; + char2 = "]"; + } + return `${char1}${label}${char2}`; + } + WithEnds() { + this.tableList = this.tableList.map((table) => { + table.Name = this.dbTypeEnds(table.Name); + table.Properties = table.Properties.map((property) => { + property.Name = this.dbTypeEnds(property.Name); + property.TableName = this.dbTypeEnds(property.TableName); + return property; + }); + return table; + }); + this.primaryKeyList = this.primaryKeyList.map((primaryKey) => { + primaryKey.PrimaryKeyName = this.dbTypeEnds(primaryKey.PrimaryKeyName); + primaryKey.PrimaryKeyTableName = this.dbTypeEnds(primaryKey.PrimaryKeyTableName); + return primaryKey; + }); + this.foreignKeyList = this.foreignKeyList.map((foreignKey) => { + foreignKey.PrimaryKeyName = this.dbTypeEnds(foreignKey.PrimaryKeyName); + foreignKey.ReferencesPropertyName = this.dbTypeEnds(foreignKey.ReferencesPropertyName); + foreignKey.PrimaryKeyTableName = this.dbTypeEnds(foreignKey.PrimaryKeyTableName); + foreignKey.ReferencesTableName = this.dbTypeEnds(foreignKey.ReferencesTableName); + return foreignKey; + }); + return this; + } + WithoutEnds() { + this.tableList.map((table) => { + table.Name = this.RemoveNameQuantifiers(table.Name); + table.Properties = table.Properties.map((property) => { + property.Name = this.RemoveNameQuantifiers(property.Name); + property.TableName = this.RemoveNameQuantifiers(property.TableName); + return property; + }); + return table; + }); + this.primaryKeyList = this.primaryKeyList.map((primaryKey) => { + primaryKey.PrimaryKeyName = this.RemoveNameQuantifiers(primaryKey.PrimaryKeyName); + primaryKey.PrimaryKeyTableName = this.RemoveNameQuantifiers(primaryKey.PrimaryKeyTableName); + return primaryKey; + }); + this.foreignKeyList = this.foreignKeyList.map((foreignKey) => { + foreignKey.PrimaryKeyName = this.RemoveNameQuantifiers(foreignKey.PrimaryKeyName); + foreignKey.ReferencesPropertyName = this.RemoveNameQuantifiers(foreignKey.ReferencesPropertyName); + foreignKey.PrimaryKeyTableName = this.RemoveNameQuantifiers(foreignKey.PrimaryKeyTableName); + foreignKey.ReferencesTableName = this.RemoveNameQuantifiers(foreignKey.ReferencesTableName); + return foreignKey; + }); + return this; + } + /** + * return text quantifiers for dialect + * @returns json + */ + GetColumnQuantifiers() { + const chars = { + Start: "\"", + End: "\"", + }; + if (this.dialect == "mysql") { + chars.Start = "`"; + chars.End = "`"; + } + else if (this.dialect == "sqlserver") { + chars.Start = "["; + chars.End = "]"; + } + return chars; + } + ToTableList() { + return this.tableList; + } + ToPrimaryKeyList() { + return this.primaryKeyList; + } + ToForeignKeyList() { + return this.foreignKeyList; + } + ResetModel() { + this.tableList = []; + this.primaryKeyList = []; + this.foreignKeyList = []; + return this; + } + /** + * return full sql model + * @returns + */ + ToModel() { + return { + TableList: this.tableList, + Dialect: this.dialect, + ForeignKeyList: this.foreignKeyList, + PrimaryKeyList: this.primaryKeyList, + }; + } + } + + //Table Info + var foreignKeyList = []; + var primaryKeyList = []; + var tableList = []; + var cells = []; + var tableCell = null; + var rowCell = null; + var dx = 0; + var exportedTables = 0; + + + //Create Base div + const divFromSQL = document.createElement('div'); + divFromSQL.style.userSelect = 'none'; + divFromSQL.style.overflow = 'hidden'; + divFromSQL.style.padding = '10px'; + divFromSQL.style.height = '100%'; + + var graph = ui.editor.graph; + + const sqlInputFromSQL = document.createElement('textarea'); + sqlInputFromSQL.style.height = '200px'; + sqlInputFromSQL.style.width = '100%'; + var defaultReset = '/*\n\tDraw io default value\n*/\n\nCREATE TABLE Persons\n(\n PersonID int NOT NULL,\n LastName varchar(255),\n ' + + 'FirstName varchar(255),\n Address varchar(255),\n City varchar(255),\n Primary Key(PersonID)\n);\n\n' + + 'CREATE TABLE Orders\n(\n OrderID int NOT NULL PRIMARY KEY,\n PersonID int NOT NULL,\n FOREIGN KEY ([PersonID]) REFERENCES [Persons]([PersonID])' + + '\n);' + sqlInputFromSQL.value = defaultReset + mxUtils.br(divFromSQL); + divFromSQL.appendChild(sqlInputFromSQL); + + var graph = ui.editor.graph; + + // Extends Extras menu + mxResources.parse('fromSql=From SQL'); + + const wndFromSQL = new mxWindow(mxResources.get('fromSql'), divFromSQL, document.body.offsetWidth - 480, 140, + 320, 320, true, true); + wndFromSQL.destroyOnClose = false; + wndFromSQL.setMaximizable(false); + wndFromSQL.setResizable(false); + wndFromSQL.setClosable(true); + + function AddRow(propertyModel, tableName) { + + var cellName = propertyModel.Name + (propertyModel.ColumnProperties ? " " + propertyModel.ColumnProperties: ""); + + rowCell = new mxCell(cellName, new mxGeometry(0, 0, 90, 26), + 'shape=partialRectangle;top=0;left=0;right=0;bottom=0;align=left;verticalAlign=top;spacingTop=-2;fillColor=none;spacingLeft=64;spacingRight=4;overflow=hidden;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;dropTarget=0;'); + rowCell.vertex = true; + + var columnType = propertyModel.IsPrimaryKey && propertyModel.IsForeignKey ? 'PK | FK' : propertyModel.IsPrimaryKey ? 'PK' : propertyModel.IsForeignKey ? 'FK' : ''; + + var left = sb.cloneCell(rowCell, columnType); + left.connectable = false; + left.style = 'shape=partialRectangle;top=0;left=0;bottom=0;fillColor=none;align=left;verticalAlign=middle;spacingLeft=4;spacingRight=4;overflow=hidden;rotatable=180;points=[];portConstraint=eastwest;part=1;' + left.geometry.width = 54; + left.geometry.height = 26; + rowCell.insert(left); + + var size = ui.editor.graph.getPreferredSizeForCell(rowCell); + + if (size !== null && tableCell.geometry.width < size.width + 10) { + tableCell.geometry.width = size.width + 10; + } + + tableCell.insert(rowCell); + tableCell.geometry.height += 26; + + rowCell = rowCell; + + }; + + function parseSql(text, type) { + // reset values + cells = [] + tableCell = null; + rowCell = null; + // load parser + const parser = new SqlSimpleParser(type); + + + const models = parser + .feed(text) + .WithoutEnds() + .WithEnds() + .ToModel(); + + + + foreignKeyList = models.ForeignKeyList; + primaryKeyList = models.PrimaryKeyList; + tableList = models.TableList; + exportedTables = tableList.length; + + //Create Table in UI + CreateTableUI(type); + }; + /** + * return text quantifiers for dialect + * @returns json + */ + function GetColumnQuantifiers(type) { + let chars = { + Start: '"', + End: '"', + }; + if (type == "mysql") { + chars.Start = "`"; + chars.End = "`"; + } + else if (type == "sqlserver") { + chars.Start = "["; + chars.End = "]"; + } + return chars; + } + + /** + * extract row column attributes + * @param {*} label + * @param {*} columnQuantifiers + * @returns + */ + function getDbLabel(label, columnQuantifiers){ + // fix duplicate spaces and different space chars + label = label + .replace(/\s+/g, " ") + let firstSpaceIndex = label[0] == columnQuantifiers.Start && + label.indexOf(columnQuantifiers.End + " ") !== -1 + ? label.indexOf(columnQuantifiers.End + " ") + : label.indexOf(" "); + let attributeType = label.substring(firstSpaceIndex + 1).trim(); + let attributeName = label.substring(0, firstSpaceIndex + 1); + let attribute = { + attributeName, + attributeType + } + return attribute + } + + function CreateTableUI(type) { + tableList.forEach(function(tableModel) { + //Define table size width + var maxNameLenght = 100 + tableModel.Name.length; + + //Create Table + tableCell = new mxCell(tableModel.Name, new mxGeometry(dx, 0, maxNameLenght, 26), + 'swimlane;fontStyle=0;childLayout=stackLayout;horizontal=1;startSize=26;fillColor=default;horizontalStack=0;resizeParent=1;resizeLast=0;collapsible=1;marginBottom=0;swimlaneFillColor=default;align=center;'); + tableCell.vertex = true; + + //Resize row + var size = ui.editor.graph.getPreferredSizeForCell(rowCell); + if (size !== null) { + tableCell.geometry.width = size.width + maxNameLenght; + } + + //Add Table to cells + cells.push(tableCell); + + //Add properties + tableModel.Properties.forEach(function(propertyModel) { + + //Add row + AddRow(propertyModel, tableModel.Name); + }); + + //Close table + dx += tableCell.geometry.width + 40; + tableCell = null; + }); + + if (cells.length > 0) { + var graph = ui.editor.graph; + var view = graph.view; + var bds = graph.getGraphBounds(); + + // Computes unscaled, untranslated graph bounds + var x = Math.ceil(Math.max(0, bds.x / view.scale - view.translate.x) + 4 * graph.gridSize); + var y = Math.ceil(Math.max(0, (bds.y + bds.height) / view.scale - view.translate.y) + 4 * graph.gridSize); + + graph.setSelectionCells(graph.importCells(cells, x, y)); + // add foreign key edges + var model = graph.getModel(); + const columnQuantifiers = GetColumnQuantifiers(type); + var pt = graph.getFreeInsertPoint(); + foreignKeyList.forEach(function(fk){ + if(fk.IsDestination && fk.PrimaryKeyName && fk.ReferencesPropertyName && + fk.PrimaryKeyTableName && fk.ReferencesTableName) { + var insertEdge = mxUtils.bind(this, function(targetCell, sourceCell, edge){ + var label = "" + var edgeStyle = "edgeStyle=entityRelationEdgeStyle;html=1;endArrow=ERzeroToMany;startArrow=ERzeroToOne;labelBackgroundColor=none;fontFamily=Verdana;fontSize=14;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=-0.018;entryY=0.608;entryDx=0;entryDy=0;entryPerimeter=0;" + var edgeCell = graph.insertEdge(null, null, label || '', (edge.invert) ? + sourceCell : targetCell, (edge.invert) ? targetCell : sourceCell, edgeStyle); + }); + let edge = { + invert: true + }; + var targetCell = null; + var sourceCell = null; + // locate edge source and target cells + for (const key in model.cells) { + if(targetCell && sourceCell) + break; + if (Object.hasOwnProperty.call(model.cells, key)) { + const mxcell = model.cells[key]; + if(mxcell.style && mxcell.style.trim().startsWith("swimlane;")){ + let entity = { + name: mxcell.value, + attributes: [] + } + var isPrimaryTable = entity.name == fk.PrimaryKeyTableName; + var isForeignTable = entity.name == fk.ReferencesTableName; + if(isPrimaryTable || isForeignTable){ + for (let c = 0; c < mxcell.children.length; c++) { + if(targetCell && sourceCell) + break; + const col = mxcell.children[c]; + if(col.mxObjectId.indexOf("mxCell") !== -1) { + if(col.style && col.style.trim().startsWith("shape=partialRectangle")){ + let attribute = getDbLabel(col.value, columnQuantifiers) + if(isPrimaryTable && attribute.attributeName == fk.PrimaryKeyName){ + targetCell = col; + break; + } else if(isForeignTable && attribute.attributeName == fk.ReferencesPropertyName){ + sourceCell = col; + break; + } + } + } + } + } + + } + } + } + if(targetCell && sourceCell) + insertEdge(targetCell, sourceCell, edge); + } + }) + graph.scrollCellToVisible(graph.getSelectionCell()); + } + + wndFromSQL.setVisible(false); + }; + + mxUtils.br(divFromSQL); + + const resetBtnFromSQL = mxUtils.button(mxResources.get('reset'), function() { + sqlInputFromSQL.value = defaultReset; + }); + + resetBtnFromSQL.style.marginTop = '8px'; + resetBtnFromSQL.style.marginRight = '4px'; + resetBtnFromSQL.style.padding = '4px'; + divFromSQL.appendChild(resetBtnFromSQL); + + const btnFromSQL_mysql = mxUtils.button('Insert MySQL', function() { + parseSql(sqlInputFromSQL.value, 'mysql'); + }); + + btnFromSQL_mysql.style.marginTop = '8px'; + btnFromSQL_mysql.style.padding = '4px'; + divFromSQL.appendChild(btnFromSQL_mysql); + + const btnFromSQL_sqlserver = mxUtils.button('Insert SQL Server', function() { + parseSql(sqlInputFromSQL.value, 'sqlserver'); + }); + + btnFromSQL_sqlserver.style.marginTop = '8px'; + btnFromSQL_sqlserver.style.padding = '4px'; + divFromSQL.appendChild(btnFromSQL_sqlserver); + + const btnFromSQL_postgres = mxUtils.button('Insert PostgreSQL', function() { + parseSql(sqlInputFromSQL.value, 'postgres'); + }); + + btnFromSQL_postgres.style.marginTop = '8px'; + btnFromSQL_postgres.style.padding = '4px'; + divFromSQL.appendChild(btnFromSQL_postgres); + + const btnFromSQL_sqlite = mxUtils.button('Insert Sqlite', function() { + parseSql(sqlInputFromSQL.value, 'sqlite'); + }); + + btnFromSQL_sqlite.style.marginTop = '8px'; + btnFromSQL_sqlite.style.padding = '4px'; + divFromSQL.appendChild(btnFromSQL_sqlite); + + // Adds action + ui.actions.addAction('fromSql', function() { + wndFromSQL.setVisible(!wndFromSQL.isVisible()); + + if (wndFromSQL.isVisible()) { + sqlInputFromSQL.focus(); + } + }); + // end import diagrams from sql text methods + + // finalize menu buttons + var theMenu = ui.menus.get('insert'); + var oldMenu = theMenu.funct; + + theMenu.funct = function(menu, parent) { + oldMenu.apply(this, arguments); + ui.menus.addMenuItems(menu, ['fromSql'], parent); + }; + if(theMenuExportAs && theMenuExportAs.enabled) { + var oldMenuExportAs = theMenuExportAs.funct; + + theMenuExportAs.funct = function(menu, parent) { + oldMenuExportAs.apply(this, arguments); + ui.menus.addMenuItems(menu, ['tosql'], parent); + }; + } else { + // vscode file export sql menu + var menu = ui.menus.get('file'); + var oldMenuExportAs = menu.funct; + menu.funct = function(menu, parent) { + oldMenuExportAs.apply(this, arguments); + debugger + ui.menus.addMenuItems(menu, ['tosql'], parent); + }; + } +}); \ No newline at end of file diff --git a/jest.config.ts b/jest.config.ts new file mode 100644 index 0000000..441575f --- /dev/null +++ b/jest.config.ts @@ -0,0 +1,28 @@ +import type { Config } from "jest"; + +const config: Config = { + verbose: true, + globals: { + "ts-jest": { + tsConfig: "tsconfig.json", + }, + }, + moduleFileExtensions: ["ts", "js"], + transform: { + "^.+\\.(ts|tsx)$": "ts-jest", + }, + testMatch: ["**/tests/**/*.spec.(ts|js)"], + // runner: "jest-serial-runner", + // testEnvironment: "node", + reporters: ["default", "jest-junit"], + collectCoverageFrom: ["src/**/*.{ts,js}", "!**/node_modules/**", "!**/lib/**"] +}; + +export default config; + +// module.exports = { +// // [...] +// // Replace `ts-jest` with the preset you want to use +// // from the above list +// preset: 'ts-jest' +// }; diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..eb68d9a --- /dev/null +++ b/package-lock.json @@ -0,0 +1,6031 @@ +{ + "name": "sqltooling-drawio", + "version": "0.0.1", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "sqltooling-drawio", + "version": "0.0.1", + "license": "MIT", + "dependencies": { + "@funktechno/little-mermaid-2-the-sql": "^0.1.1", + "ts-node": "^9.0.0" + }, + "devDependencies": { + "jest": "^29.0.3", + "jest-junit": "^16.0.0", + "jest-serial-runner": "^1.1.0", + "ts-jest": "^29.0.1", + "ts-node": "^10.9.1", + "typescript": "^5.0.2" + }, + "engines": { + "node": ">=16.x" + } + }, + "node_modules/@ampproject/remapping": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.3.0.tgz", + "integrity": "sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==", + "dev": true, + "dependencies": { + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/code-frame": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.24.7.tgz", + "integrity": "sha512-BcYH1CVJBO9tvyIZ2jVeXgSIMvGZ2FDRvDdOIVQyuklNKSsx+eppDEBq/g47Ayw+RqNFE+URvOShmf+f/qwAlA==", + "dev": true, + "dependencies": { + "@babel/highlight": "^7.24.7", + "picocolors": "^1.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/compat-data": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.24.7.tgz", + "integrity": "sha512-qJzAIcv03PyaWqxRgO4mSU3lihncDT296vnyuE2O8uA4w3UHWI4S3hgeZd1L8W1Bft40w9JxJ2b412iDUFFRhw==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.24.7.tgz", + "integrity": "sha512-nykK+LEK86ahTkX/3TgauT0ikKoNCfKHEaZYTUVupJdTLzGNvrblu4u6fa7DhZONAltdf8e662t/abY8idrd/g==", + "dev": true, + "dependencies": { + "@ampproject/remapping": "^2.2.0", + "@babel/code-frame": "^7.24.7", + "@babel/generator": "^7.24.7", + "@babel/helper-compilation-targets": "^7.24.7", + "@babel/helper-module-transforms": "^7.24.7", + "@babel/helpers": "^7.24.7", + "@babel/parser": "^7.24.7", + "@babel/template": "^7.24.7", + "@babel/traverse": "^7.24.7", + "@babel/types": "^7.24.7", + "convert-source-map": "^2.0.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.3", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" + } + }, + "node_modules/@babel/core/node_modules/convert-source-map": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", + "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", + "dev": true + }, + "node_modules/@babel/core/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, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/generator": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.24.7.tgz", + "integrity": "sha512-oipXieGC3i45Y1A41t4tAqpnEZWgB/lC6Ehh6+rOviR5XWpTtMmLN+fGjz9vOiNRt0p6RtO6DtD0pdU3vpqdSA==", + "dev": true, + "dependencies": { + "@babel/types": "^7.24.7", + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.25", + "jsesc": "^2.5.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.24.7.tgz", + "integrity": "sha512-ctSdRHBi20qWOfy27RUb4Fhp07KSJ3sXcuSvTrXrc4aG8NSYDo1ici3Vhg9bg69y5bj0Mr1lh0aeEgTvc12rMg==", + "dev": true, + "dependencies": { + "@babel/compat-data": "^7.24.7", + "@babel/helper-validator-option": "^7.24.7", + "browserslist": "^4.22.2", + "lru-cache": "^5.1.1", + "semver": "^6.3.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets/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, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/helper-environment-visitor": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-environment-visitor/-/helper-environment-visitor-7.24.7.tgz", + "integrity": "sha512-DoiN84+4Gnd0ncbBOM9AZENV4a5ZiL39HYMyZJGZ/AZEykHYdJw0wW3kdcsh9/Kn+BRXHLkkklZ51ecPKmI1CQ==", + "dev": true, + "dependencies": { + "@babel/types": "^7.24.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-function-name": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.24.7.tgz", + "integrity": "sha512-FyoJTsj/PEUWu1/TYRiXTIHc8lbw+TDYkZuoE43opPS5TrI7MyONBE1oNvfguEXAD9yhQRrVBnXdXzSLQl9XnA==", + "dev": true, + "dependencies": { + "@babel/template": "^7.24.7", + "@babel/types": "^7.24.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-hoist-variables": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.24.7.tgz", + "integrity": "sha512-MJJwhkoGy5c4ehfoRyrJ/owKeMl19U54h27YYftT0o2teQ3FJ3nQUf/I3LlJsX4l3qlw7WRXUmiyajvHXoTubQ==", + "dev": true, + "dependencies": { + "@babel/types": "^7.24.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-imports": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.24.7.tgz", + "integrity": "sha512-8AyH3C+74cgCVVXow/myrynrAGv+nTVg5vKu2nZph9x7RcRwzmh0VFallJuFTZ9mx6u4eSdXZfcOzSqTUm0HCA==", + "dev": true, + "dependencies": { + "@babel/traverse": "^7.24.7", + "@babel/types": "^7.24.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-transforms": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.24.7.tgz", + "integrity": "sha512-1fuJEwIrp+97rM4RWdO+qrRsZlAeL1lQJoPqtCYWv0NL115XM93hIH4CSRln2w52SqvmY5hqdtauB6QFCDiZNQ==", + "dev": true, + "dependencies": { + "@babel/helper-environment-visitor": "^7.24.7", + "@babel/helper-module-imports": "^7.24.7", + "@babel/helper-simple-access": "^7.24.7", + "@babel/helper-split-export-declaration": "^7.24.7", + "@babel/helper-validator-identifier": "^7.24.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-plugin-utils": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.24.7.tgz", + "integrity": "sha512-Rq76wjt7yz9AAc1KnlRKNAi/dMSVWgDRx43FHoJEbcYU6xOWaE2dVPwcdTukJrjxS65GITyfbvEYHvkirZ6uEg==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-simple-access": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.24.7.tgz", + "integrity": "sha512-zBAIvbCMh5Ts+b86r/CjU+4XGYIs+R1j951gxI3KmmxBMhCg4oQMsv6ZXQ64XOm/cvzfU1FmoCyt6+owc5QMYg==", + "dev": true, + "dependencies": { + "@babel/traverse": "^7.24.7", + "@babel/types": "^7.24.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-split-export-declaration": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.24.7.tgz", + "integrity": "sha512-oy5V7pD+UvfkEATUKvIjvIAH/xCzfsFVw7ygW2SI6NClZzquT+mwdTfgfdbUiceh6iQO0CHtCPsyze/MZ2YbAA==", + "dev": true, + "dependencies": { + "@babel/types": "^7.24.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-string-parser": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.24.7.tgz", + "integrity": "sha512-7MbVt6xrwFQbunH2DNQsAP5sTGxfqQtErvBIvIMi6EQnbgUOuVYanvREcmFrOPhoXBrTtjhhP+lW+o5UfK+tDg==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.24.7.tgz", + "integrity": "sha512-rR+PBcQ1SMQDDyF6X0wxtG8QyLCgUB0eRAGguqRLfkCA87l7yAP7ehq8SNj96OOGTO8OBV70KhuFYcIkHXOg0w==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.24.7.tgz", + "integrity": "sha512-yy1/KvjhV/ZCL+SM7hBrvnZJ3ZuT9OuZgIJAGpPEToANvc3iM6iDvBnRjtElWibHU6n8/LPR/EjX9EtIEYO3pw==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.24.7.tgz", + "integrity": "sha512-NlmJJtvcw72yRJRcnCmGvSi+3jDEg8qFu3z0AFoymmzLx5ERVWyzd9kVXr7Th9/8yIJi2Zc6av4Tqz3wFs8QWg==", + "dev": true, + "dependencies": { + "@babel/template": "^7.24.7", + "@babel/types": "^7.24.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/highlight": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.24.7.tgz", + "integrity": "sha512-EStJpq4OuY8xYfhGVXngigBJRWxftKX9ksiGDnmlY3o7B/V7KIAc9X4oiK87uPJSc/vs5L869bem5fhZa8caZw==", + "dev": true, + "dependencies": { + "@babel/helper-validator-identifier": "^7.24.7", + "chalk": "^2.4.2", + "js-tokens": "^4.0.0", + "picocolors": "^1.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/highlight/node_modules/ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "dependencies": { + "color-convert": "^1.9.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/@babel/highlight/node_modules/chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dev": true, + "dependencies": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/@babel/highlight/node_modules/color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dev": true, + "dependencies": { + "color-name": "1.1.3" + } + }, + "node_modules/@babel/highlight/node_modules/color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", + "dev": true + }, + "node_modules/@babel/highlight/node_modules/has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/@babel/highlight/node_modules/supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "dependencies": { + "has-flag": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/@babel/parser": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.24.7.tgz", + "integrity": "sha512-9uUYRm6OqQrCqQdG1iCBwBPZgN8ciDBro2nIOFaiRz1/BCxaI7CNvQbDHvsArAC7Tw9Hda/B3U+6ui9u4HWXPw==", + "dev": true, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/plugin-syntax-async-generators": { + "version": "7.8.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz", + "integrity": "sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-bigint": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-bigint/-/plugin-syntax-bigint-7.8.3.tgz", + "integrity": "sha512-wnTnFlG+YxQm3vDxpGE57Pj0srRU4sHE/mDkt1qv2YJJSeUAec2ma4WLUnUPeKjyrfntVwe/N6dCXpU+zL3Npg==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-class-properties": { + "version": "7.12.13", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.12.13.tgz", + "integrity": "sha512-fm4idjKla0YahUNgFNLCB0qySdsoPiZP3iQE3rky0mBUtMZ23yDJ9SJdg6dXTSDnulOVqiF3Hgr9nbXvXTQZYA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.12.13" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-import-meta": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-import-meta/-/plugin-syntax-import-meta-7.10.4.tgz", + "integrity": "sha512-Yqfm+XDx0+Prh3VSeEQCPU81yC+JWZ2pDPFSS4ZdpfZhp4MkFMaDC1UqseovEKwSUpnIL7+vK+Clp7bfh0iD7g==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-json-strings": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz", + "integrity": "sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-jsx": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-jsx/-/plugin-syntax-jsx-7.24.7.tgz", + "integrity": "sha512-6ddciUPe/mpMnOKv/U+RSd2vvVy+Yw/JfBB0ZHYjEZt9NLHmCUylNYlsbqCCS1Bffjlb0fCwC9Vqz+sBz6PsiQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.24.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-logical-assignment-operators": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-logical-assignment-operators/-/plugin-syntax-logical-assignment-operators-7.10.4.tgz", + "integrity": "sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-nullish-coalescing-operator": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-nullish-coalescing-operator/-/plugin-syntax-nullish-coalescing-operator-7.8.3.tgz", + "integrity": "sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-numeric-separator": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.10.4.tgz", + "integrity": "sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-object-rest-spread": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz", + "integrity": "sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-catch-binding": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz", + "integrity": "sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-optional-chaining": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-chaining/-/plugin-syntax-optional-chaining-7.8.3.tgz", + "integrity": "sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-top-level-await": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.14.5.tgz", + "integrity": "sha512-hx++upLv5U1rgYfwe1xBQUhRmU41NEvpUvrp8jkrSCdvGSnM5/qdRMtylJ6PG5OFkBaHkbTAKTnd3/YyESRHFw==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-typescript": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-typescript/-/plugin-syntax-typescript-7.24.7.tgz", + "integrity": "sha512-c/+fVeJBB0FeKsFvwytYiUD+LBvhHjGSI0g446PRGdSVGZLRNArBUno2PETbAly3tpiNAQR5XaZ+JslxkotsbA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.24.7" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/template": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.24.7.tgz", + "integrity": "sha512-jYqfPrU9JTF0PmPy1tLYHW4Mp4KlgxJD9l2nP9fD6yT/ICi554DmrWBAEYpIelzjHf1msDP3PxJIRt/nFNfBig==", + "dev": true, + "dependencies": { + "@babel/code-frame": "^7.24.7", + "@babel/parser": "^7.24.7", + "@babel/types": "^7.24.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.24.7.tgz", + "integrity": "sha512-yb65Ed5S/QAcewNPh0nZczy9JdYXkkAbIsEo+P7BE7yO3txAY30Y/oPa3QkQ5It3xVG2kpKMg9MsdxZaO31uKA==", + "dev": true, + "dependencies": { + "@babel/code-frame": "^7.24.7", + "@babel/generator": "^7.24.7", + "@babel/helper-environment-visitor": "^7.24.7", + "@babel/helper-function-name": "^7.24.7", + "@babel/helper-hoist-variables": "^7.24.7", + "@babel/helper-split-export-declaration": "^7.24.7", + "@babel/parser": "^7.24.7", + "@babel/types": "^7.24.7", + "debug": "^4.3.1", + "globals": "^11.1.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse/node_modules/globals": { + "version": "11.12.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", + "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/@babel/types": { + "version": "7.24.7", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.24.7.tgz", + "integrity": "sha512-XEFXSlxiG5td2EJRe8vOmRbaXVgfcBlszKujvVmWIK/UpywWljQCfzAv3RQCGujWQ1RD4YYWEAqDXfuJiy8f5Q==", + "dev": true, + "dependencies": { + "@babel/helper-string-parser": "^7.24.7", + "@babel/helper-validator-identifier": "^7.24.7", + "to-fast-properties": "^2.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@bcoe/v8-coverage": { + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/@bcoe/v8-coverage/-/v8-coverage-0.2.3.tgz", + "integrity": "sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==", + "dev": true + }, + "node_modules/@braintree/sanitize-url": { + "version": "6.0.4", + "resolved": "https://registry.npmjs.org/@braintree/sanitize-url/-/sanitize-url-6.0.4.tgz", + "integrity": "sha512-s3jaWicZd0pkP0jf5ysyHUI/RE7MHos6qlToFcGWXVp+ykHOy77OUMrfbgJ9it2C5bow7OIQwYYaHjk9XlBQ2A==" + }, + "node_modules/@cspotcode/source-map-support": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/@cspotcode/source-map-support/-/source-map-support-0.8.1.tgz", + "integrity": "sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==", + "dev": true, + "dependencies": { + "@jridgewell/trace-mapping": "0.3.9" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/@cspotcode/source-map-support/node_modules/@jridgewell/trace-mapping": { + "version": "0.3.9", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz", + "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==", + "dev": true, + "dependencies": { + "@jridgewell/resolve-uri": "^3.0.3", + "@jridgewell/sourcemap-codec": "^1.4.10" + } + }, + "node_modules/@funktechno/little-mermaid-2-the-sql": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/@funktechno/little-mermaid-2-the-sql/-/little-mermaid-2-the-sql-0.1.1.tgz", + "integrity": "sha512-BuWX/8sm2N5Atj2K8B+e5JIUWOt/5wib5LrAtShjq8cyX3BHi4ftGajE9pE/EKxY2Idj1FcOZbUGWwSuASN+iQ==", + "dependencies": { + "@types/dompurify": "^3.0.0", + "chalk": "^5.1.2", + "clear": "^0.1.0", + "commander": "^11.0.0", + "figlet": "^1.5.0", + "jison": "^0.4.18", + "lodash": "^4.17.21", + "mermaid": "^10.0.0", + "moment-mini": "^2.24.0", + "path": "^0.12.7" + }, + "bin": { + "littleMermaid2SQL": "lib/src/index.js" + }, + "engines": { + "node": ">=16.x" + } + }, + "node_modules/@istanbuljs/load-nyc-config": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@istanbuljs/load-nyc-config/-/load-nyc-config-1.1.0.tgz", + "integrity": "sha512-VjeHSlIzpv/NyD3N0YuHfXOPDIixcA1q2ZV98wsMqcYlPmv2n3Yb2lYP9XMElnaFVXg5A7YLTeLu6V84uQDjmQ==", + "dev": true, + "dependencies": { + "camelcase": "^5.3.1", + "find-up": "^4.1.0", + "get-package-type": "^0.1.0", + "js-yaml": "^3.13.1", + "resolve-from": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/resolve-from": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", + "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/schema": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz", + "integrity": "sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/console": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/console/-/console-29.7.0.tgz", + "integrity": "sha512-5Ni4CU7XHQi32IJ398EEP4RrB8eV09sXP2ROqD4bksHrnTree52PsxvX8tpL8LvTZ3pFzXyPbNQReSN41CAhOg==", + "dev": true, + "dependencies": { + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/console/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/@jest/core": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/core/-/core-29.7.0.tgz", + "integrity": "sha512-n7aeXWKMnGtDA48y8TLWJPJmLmmZ642Ceo78cYWEpiD7FzDgmNDV/GCVRorPABdXLJZ/9wzzgZAlHjXjxDHGsg==", + "dev": true, + "dependencies": { + "@jest/console": "^29.7.0", + "@jest/reporters": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "ansi-escapes": "^4.2.1", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "exit": "^0.1.2", + "graceful-fs": "^4.2.9", + "jest-changed-files": "^29.7.0", + "jest-config": "^29.7.0", + "jest-haste-map": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-regex-util": "^29.6.3", + "jest-resolve": "^29.7.0", + "jest-resolve-dependencies": "^29.7.0", + "jest-runner": "^29.7.0", + "jest-runtime": "^29.7.0", + "jest-snapshot": "^29.7.0", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "jest-watcher": "^29.7.0", + "micromatch": "^4.0.4", + "pretty-format": "^29.7.0", + "slash": "^3.0.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/@jest/core/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/@jest/core/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/core/node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@jest/core/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true + }, + "node_modules/@jest/core/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/environment": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/environment/-/environment-29.7.0.tgz", + "integrity": "sha512-aQIfHDq33ExsN4jP1NWGXhxgQ/wixs60gDiKO+XVMd8Mn0NWPWgc34ZQDTb2jKaUWQ7MuwoitXAsN2XVXNMpAw==", + "dev": true, + "dependencies": { + "@jest/fake-timers": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "jest-mock": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/expect": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/expect/-/expect-29.7.0.tgz", + "integrity": "sha512-8uMeAMycttpva3P1lBHB8VciS9V0XAr3GymPpipdyQXbBcuhkLQOSe8E/p92RyAdToS6ZD1tFkX+CkhoECE0dQ==", + "dev": true, + "dependencies": { + "expect": "^29.7.0", + "jest-snapshot": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/expect-utils": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/expect-utils/-/expect-utils-29.7.0.tgz", + "integrity": "sha512-GlsNBWiFQFCVi9QVSx7f5AgMeLxe9YCCs5PuP2O2LdjDAA8Jh9eX7lA1Jq/xdXw3Wb3hyvlFNfZIfcRetSzYcA==", + "dev": true, + "dependencies": { + "jest-get-type": "^29.6.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/expect-utils/node_modules/jest-get-type": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", + "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "dev": true, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/fake-timers": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/fake-timers/-/fake-timers-29.7.0.tgz", + "integrity": "sha512-q4DH1Ha4TTFPdxLsqDXK1d3+ioSL7yL5oCMJZgDYm6i+6CygW5E5xVr/D1HdsGxjt1ZWSfUAs9OxSB/BNelWrQ==", + "dev": true, + "dependencies": { + "@jest/types": "^29.6.3", + "@sinonjs/fake-timers": "^10.0.2", + "@types/node": "*", + "jest-message-util": "^29.7.0", + "jest-mock": "^29.7.0", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/globals": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/globals/-/globals-29.7.0.tgz", + "integrity": "sha512-mpiz3dutLbkW2MNFubUGUEVLkTGiqW6yLVTA+JbP6fI6J5iL9Y0Nlg8k95pcF8ctKwCS7WVxteBs29hhfAotzQ==", + "dev": true, + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/expect": "^29.7.0", + "@jest/types": "^29.6.3", + "jest-mock": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/reporters": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/reporters/-/reporters-29.7.0.tgz", + "integrity": "sha512-DApq0KJbJOEzAFYjHADNNxAE3KbhxQB1y5Kplb5Waqw6zVbuWatSnMjE5gs8FUgEPmNsnZA3NCWl9NG0ia04Pg==", + "dev": true, + "dependencies": { + "@bcoe/v8-coverage": "^0.2.3", + "@jest/console": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@jridgewell/trace-mapping": "^0.3.18", + "@types/node": "*", + "chalk": "^4.0.0", + "collect-v8-coverage": "^1.0.0", + "exit": "^0.1.2", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "istanbul-lib-coverage": "^3.0.0", + "istanbul-lib-instrument": "^6.0.0", + "istanbul-lib-report": "^3.0.0", + "istanbul-lib-source-maps": "^4.0.0", + "istanbul-reports": "^3.1.3", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0", + "jest-worker": "^29.7.0", + "slash": "^3.0.0", + "string-length": "^4.0.1", + "strip-ansi": "^6.0.0", + "v8-to-istanbul": "^9.0.1" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/@jest/reporters/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/@jest/reporters/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@jest/schemas": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/@jest/schemas/-/schemas-29.6.3.tgz", + "integrity": "sha512-mo5j5X+jIZmJQveBKeS/clAueipV7KgiX1vMgCxam1RNYiqE1w62n0/tJJnHtjW8ZHcQco5gY85jA3mi0L+nSA==", + "dev": true, + "dependencies": { + "@sinclair/typebox": "^0.27.8" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/source-map": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/@jest/source-map/-/source-map-29.6.3.tgz", + "integrity": "sha512-MHjT95QuipcPrpLM+8JMSzFx6eHp5Bm+4XeFDJlwsvVBjmKNiIAvasGK2fxz2WbGRlnvqehFbh07MMa7n3YJnw==", + "dev": true, + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.18", + "callsites": "^3.0.0", + "graceful-fs": "^4.2.9" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/test-result": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/test-result/-/test-result-29.7.0.tgz", + "integrity": "sha512-Fdx+tv6x1zlkJPcWXmMDAG2HBnaR9XPSd5aDWQVsfrZmLVT3lU1cwyxLgRmXR9yrq4NBoEm9BMsfgFzTQAbJYA==", + "dev": true, + "dependencies": { + "@jest/console": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/istanbul-lib-coverage": "^2.0.0", + "collect-v8-coverage": "^1.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/test-sequencer": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/test-sequencer/-/test-sequencer-29.7.0.tgz", + "integrity": "sha512-GQwJ5WZVrKnOJuiYiAF52UNUJXgTZx1NHjFSEB0qEMmSZKAkdMoIzw/Cj6x6NF4AvV23AUqDpFzQkN/eYCYTxw==", + "dev": true, + "dependencies": { + "@jest/test-result": "^29.7.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/transform": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/@jest/transform/-/transform-29.7.0.tgz", + "integrity": "sha512-ok/BTPFzFKVMwO5eOHRrvnBVHdRy9IrsrW1GpMaQ9MCnilNLXQKmAX8s1YXDFaai9xJpac2ySzV0YeRRECr2Vw==", + "dev": true, + "dependencies": { + "@babel/core": "^7.11.6", + "@jest/types": "^29.6.3", + "@jridgewell/trace-mapping": "^0.3.18", + "babel-plugin-istanbul": "^6.1.1", + "chalk": "^4.0.0", + "convert-source-map": "^2.0.0", + "fast-json-stable-stringify": "^2.1.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "jest-regex-util": "^29.6.3", + "jest-util": "^29.7.0", + "micromatch": "^4.0.4", + "pirates": "^4.0.4", + "slash": "^3.0.0", + "write-file-atomic": "^4.0.2" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/transform/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/@jest/transform/node_modules/convert-source-map": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", + "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", + "dev": true + }, + "node_modules/@jest/types": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-29.6.3.tgz", + "integrity": "sha512-u3UPsIilWKOM3F9CXtrG8LEJmNxwoCQC/XVj4IKYXvvpx7QIi/Kg1LI5uDmDpKlac62NUtX7eLjRh+jVZcLOzw==", + "dev": true, + "dependencies": { + "@jest/schemas": "^29.6.3", + "@types/istanbul-lib-coverage": "^2.0.0", + "@types/istanbul-reports": "^3.0.0", + "@types/node": "*", + "@types/yargs": "^17.0.8", + "chalk": "^4.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/@jest/types/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/@jridgewell/gen-mapping": { + "version": "0.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.5.tgz", + "integrity": "sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==", + "dev": true, + "dependencies": { + "@jridgewell/set-array": "^1.2.1", + "@jridgewell/sourcemap-codec": "^1.4.10", + "@jridgewell/trace-mapping": "^0.3.24" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", + "dev": true, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/set-array": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.2.1.tgz", + "integrity": "sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==", + "dev": true, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.4.15", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz", + "integrity": "sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==", + "dev": true + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.25", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", + "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", + "dev": true, + "dependencies": { + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" + } + }, + "node_modules/@sinclair/typebox": { + "version": "0.27.8", + "resolved": "https://registry.npmjs.org/@sinclair/typebox/-/typebox-0.27.8.tgz", + "integrity": "sha512-+Fj43pSMwJs4KRrH/938Uf+uAELIgVBmQzg/q1YG10djyfA3TnrU8N8XzqCh/okZdszqBQTZf96idMfE5lnwTA==", + "dev": true + }, + "node_modules/@sinonjs/commons": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-3.0.1.tgz", + "integrity": "sha512-K3mCHKQ9sVh8o1C9cxkwxaOmXoAMlDxC1mYyHrjqOWEcBjYr76t96zL2zlj5dUGZ3HSw240X1qgH3Mjf1yJWpQ==", + "dev": true, + "dependencies": { + "type-detect": "4.0.8" + } + }, + "node_modules/@sinonjs/fake-timers": { + "version": "10.3.0", + "resolved": "https://registry.npmjs.org/@sinonjs/fake-timers/-/fake-timers-10.3.0.tgz", + "integrity": "sha512-V4BG07kuYSUkTCSBHG8G8TNhM+F19jXFWnQtzj+we8DrkpSBCee9Z3Ms8yiGer/dlmhe35/Xdgyo3/0rQKg7YA==", + "dev": true, + "dependencies": { + "@sinonjs/commons": "^3.0.0" + } + }, + "node_modules/@tsconfig/node10": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.11.tgz", + "integrity": "sha512-DcRjDCujK/kCk/cUe8Xz8ZSpm8mS3mNNpta+jGCA6USEDfktlNvm1+IuZ9eTcDbNk41BHwpHHeW+N1lKCz4zOw==", + "dev": true + }, + "node_modules/@tsconfig/node12": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/@tsconfig/node12/-/node12-1.0.11.tgz", + "integrity": "sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==", + "dev": true + }, + "node_modules/@tsconfig/node14": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@tsconfig/node14/-/node14-1.0.3.tgz", + "integrity": "sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==", + "dev": true + }, + "node_modules/@tsconfig/node16": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/@tsconfig/node16/-/node16-1.0.4.tgz", + "integrity": "sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==", + "dev": true + }, + "node_modules/@types/babel__core": { + "version": "7.20.5", + "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.20.5.tgz", + "integrity": "sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==", + "dev": true, + "dependencies": { + "@babel/parser": "^7.20.7", + "@babel/types": "^7.20.7", + "@types/babel__generator": "*", + "@types/babel__template": "*", + "@types/babel__traverse": "*" + } + }, + "node_modules/@types/babel__generator": { + "version": "7.6.8", + "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.6.8.tgz", + "integrity": "sha512-ASsj+tpEDsEiFr1arWrlN6V3mdfjRMZt6LtK/Vp/kreFLnr5QH5+DhvD5nINYZXzwJvXeGq+05iUXcAzVrqWtw==", + "dev": true, + "dependencies": { + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__template": { + "version": "7.4.4", + "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.4.tgz", + "integrity": "sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A==", + "dev": true, + "dependencies": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0" + } + }, + "node_modules/@types/babel__traverse": { + "version": "7.20.6", + "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.20.6.tgz", + "integrity": "sha512-r1bzfrm0tomOI8g1SzvCaQHo6Lcv6zu0EA+W2kHrt8dyrHQxGzBBL4kdkzIS+jBMV+EYcMAEAqXqYaLJq5rOZg==", + "dev": true, + "dependencies": { + "@babel/types": "^7.20.7" + } + }, + "node_modules/@types/d3-scale": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/@types/d3-scale/-/d3-scale-4.0.8.tgz", + "integrity": "sha512-gkK1VVTr5iNiYJ7vWDI+yUFFlszhNMtVeneJ6lUTKPjprsvLLI9/tgEGiXJOnlINJA8FyA88gfnQsHbybVZrYQ==", + "dependencies": { + "@types/d3-time": "*" + } + }, + "node_modules/@types/d3-scale-chromatic": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/@types/d3-scale-chromatic/-/d3-scale-chromatic-3.0.3.tgz", + "integrity": "sha512-laXM4+1o5ImZv3RpFAsTRn3TEkzqkytiOY0Dz0sq5cnd1dtNlk6sHLon4OvqaiJb28T0S/TdsBI3Sjsy+keJrw==" + }, + "node_modules/@types/d3-time": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/@types/d3-time/-/d3-time-3.0.3.tgz", + "integrity": "sha512-2p6olUZ4w3s+07q3Tm2dbiMZy5pCDfYwtLXXHUnVzXgQlZ/OyPtUz6OL382BkOuGlLXqfT+wqv8Fw2v8/0geBw==" + }, + "node_modules/@types/debug": { + "version": "4.1.12", + "resolved": "https://registry.npmjs.org/@types/debug/-/debug-4.1.12.tgz", + "integrity": "sha512-vIChWdVG3LG1SMxEvI/AK+FWJthlrqlTu7fbrlywTkkaONwk/UAGaULXRlf8vkzFBLVm0zkMdCquhL5aOjhXPQ==", + "dependencies": { + "@types/ms": "*" + } + }, + "node_modules/@types/dompurify": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@types/dompurify/-/dompurify-3.0.5.tgz", + "integrity": "sha512-1Wg0g3BtQF7sSb27fJQAKck1HECM6zV1EB66j8JH9i3LCjYabJa0FSdiSgsD5K/RbrsR0SiraKacLB+T8ZVYAg==", + "dependencies": { + "@types/trusted-types": "*" + } + }, + "node_modules/@types/graceful-fs": { + "version": "4.1.9", + "resolved": "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.9.tgz", + "integrity": "sha512-olP3sd1qOEe5dXTSaFvQG+02VdRXcdytWLAZsAq1PecU8uqQAhkrnbli7DagjtXKW/Bl7YJbUsa8MPcuc8LHEQ==", + "dev": true, + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/istanbul-lib-coverage": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.6.tgz", + "integrity": "sha512-2QF/t/auWm0lsy8XtKVPG19v3sSOQlJe/YHZgfjb/KBBHOGSV+J2q/S671rcq9uTBrLAXmZpqJiaQbMT+zNU1w==", + "dev": true + }, + "node_modules/@types/istanbul-lib-report": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/@types/istanbul-lib-report/-/istanbul-lib-report-3.0.3.tgz", + "integrity": "sha512-NQn7AHQnk/RSLOxrBbGyJM/aVQ+pjj5HCgasFxc0K/KhoATfQ/47AyUl15I2yBUpihjmas+a+VJBOqecrFH+uA==", + "dev": true, + "dependencies": { + "@types/istanbul-lib-coverage": "*" + } + }, + "node_modules/@types/istanbul-reports": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.4.tgz", + "integrity": "sha512-pk2B1NWalF9toCRu6gjBzR69syFjP4Od8WRAX+0mmf9lAjCRicLOWc+ZrxZHx/0XRjotgkF9t6iaMJ+aXcOdZQ==", + "dev": true, + "dependencies": { + "@types/istanbul-lib-report": "*" + } + }, + "node_modules/@types/mdast": { + "version": "3.0.15", + "resolved": "https://registry.npmjs.org/@types/mdast/-/mdast-3.0.15.tgz", + "integrity": "sha512-LnwD+mUEfxWMa1QpDraczIn6k0Ee3SMicuYSSzS6ZYl2gKS09EClnJYGd8Du6rfc5r/GZEk5o1mRb8TaTj03sQ==", + "dependencies": { + "@types/unist": "^2" + } + }, + "node_modules/@types/ms": { + "version": "0.7.34", + "resolved": "https://registry.npmjs.org/@types/ms/-/ms-0.7.34.tgz", + "integrity": "sha512-nG96G3Wp6acyAgJqGasjODb+acrI7KltPiRxzHPXnP3NgI28bpQDRv53olbqGXbfcgF5aiiHmO3xpwEpS5Ld9g==" + }, + "node_modules/@types/node": { + "version": "14.18.63", + "resolved": "https://registry.npmjs.org/@types/node/-/node-14.18.63.tgz", + "integrity": "sha512-fAtCfv4jJg+ExtXhvCkCqUKZ+4ok/JQk01qDKhL5BDDoS3AxKXhV5/MAVUZyQnSEd2GT92fkgZl0pz0Q0AzcIQ==", + "dev": true + }, + "node_modules/@types/stack-utils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.3.tgz", + "integrity": "sha512-9aEbYZ3TbYMznPdcdr3SmIrLXwC/AKZXQeCf9Pgao5CKb8CyHuEX5jzWPTkvregvhRJHcpRO6BFoGW9ycaOkYw==", + "dev": true + }, + "node_modules/@types/trusted-types": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/@types/trusted-types/-/trusted-types-2.0.7.tgz", + "integrity": "sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw==" + }, + "node_modules/@types/unist": { + "version": "2.0.10", + "resolved": "https://registry.npmjs.org/@types/unist/-/unist-2.0.10.tgz", + "integrity": "sha512-IfYcSBWE3hLpBg8+X2SEa8LVkJdJEkT2Ese2aaLs3ptGdVtABxndrMaxuFlQ1qdFf9Q5rDvDpxI3WwgvKFAsQA==" + }, + "node_modules/@types/yargs": { + "version": "17.0.32", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.32.tgz", + "integrity": "sha512-xQ67Yc/laOG5uMfX/093MRlGGCIBzZMarVa+gfNKJxWAIgykYpVGkBdbqEzGDDfCrVUj6Hiff4mTZ5BA6TmAog==", + "dev": true, + "dependencies": { + "@types/yargs-parser": "*" + } + }, + "node_modules/@types/yargs-parser": { + "version": "21.0.3", + "resolved": "https://registry.npmjs.org/@types/yargs-parser/-/yargs-parser-21.0.3.tgz", + "integrity": "sha512-I4q9QU9MQv4oEOz4tAHJtNz1cwuLxn2F3xcc2iV5WdqLPpUnj30aUuxt1mAxYTG+oe8CZMV/+6rU4S4gRDzqtQ==", + "dev": true + }, + "node_modules/amdefine": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/amdefine/-/amdefine-1.0.1.tgz", + "integrity": "sha512-S2Hw0TtNkMJhIabBwIojKL9YHO5T0n5eNqWJ7Lrlel/zDbftQpxpapi8tZs3X1HWa+u+QeydGmzzNU0m09+Rcg==", + "optional": true, + "engines": { + "node": ">=0.4.2" + } + }, + "node_modules/ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "dependencies": { + "type-fest": "^0.21.3" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/ansi-escapes/node_modules/type-fest": { + "version": "0.21.3", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.21.3.tgz", + "integrity": "sha512-t0rzBq87m3fVcduHDUFhKmyyX+9eo6WQjZvf51Ea/M0Q7+T374Jp1aUiyUl0GKxp8M/OETVHSDvmkyPgvX+X2w==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "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==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/anymatch": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", + "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", + "dev": true, + "dependencies": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/arg": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", + "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", + "dev": true + }, + "node_modules/argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dev": true, + "dependencies": { + "sprintf-js": "~1.0.2" + } + }, + "node_modules/babel-jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/babel-jest/-/babel-jest-29.7.0.tgz", + "integrity": "sha512-BrvGY3xZSwEcCzKvKsCi2GgHqDqsYkOP4/by5xCgIwGXQxIEh+8ew3gmrE1y7XRR6LHZIj6yLYnUi/mm2KXKBg==", + "dev": true, + "dependencies": { + "@jest/transform": "^29.7.0", + "@types/babel__core": "^7.1.14", + "babel-plugin-istanbul": "^6.1.1", + "babel-preset-jest": "^29.6.3", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.8.0" + } + }, + "node_modules/babel-jest/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/babel-plugin-istanbul": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/babel-plugin-istanbul/-/babel-plugin-istanbul-6.1.1.tgz", + "integrity": "sha512-Y1IQok9821cC9onCx5otgFfRm7Lm+I+wwxOx738M/WLPZ9Q42m4IG5W0FNX8WLL2gYMZo3JkuXIH2DOpWM+qwA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.0.0", + "@istanbuljs/load-nyc-config": "^1.0.0", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-instrument": "^5.0.4", + "test-exclude": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-plugin-istanbul/node_modules/istanbul-lib-instrument": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-5.2.1.tgz", + "integrity": "sha512-pzqtp31nLv/XFOzXGuvhCb8qhjmTVo5vjVk19XE4CRlSWz0KoeJ3bw9XsA7nOp9YBf4qHjwBxkDzKcME/J29Yg==", + "dev": true, + "dependencies": { + "@babel/core": "^7.12.3", + "@babel/parser": "^7.14.7", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-coverage": "^3.2.0", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/babel-plugin-istanbul/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, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/babel-plugin-jest-hoist": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-29.6.3.tgz", + "integrity": "sha512-ESAc/RJvGTFEzRwOTT4+lNDk/GNHMkKbNzsvT0qKRfDyyYTskxB5rnU2njIDYVxXCBHHEI1c0YwHob3WaYujOg==", + "dev": true, + "dependencies": { + "@babel/template": "^7.3.3", + "@babel/types": "^7.3.3", + "@types/babel__core": "^7.1.14", + "@types/babel__traverse": "^7.0.6" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/babel-preset-current-node-syntax": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.0.1.tgz", + "integrity": "sha512-M7LQ0bxarkxQoN+vz5aJPsLBn77n8QgTFmo8WK0/44auK2xlCXrYcUxHFxgU7qW5Yzw/CjmLRK2uJzaCd7LvqQ==", + "dev": true, + "dependencies": { + "@babel/plugin-syntax-async-generators": "^7.8.4", + "@babel/plugin-syntax-bigint": "^7.8.3", + "@babel/plugin-syntax-class-properties": "^7.8.3", + "@babel/plugin-syntax-import-meta": "^7.8.3", + "@babel/plugin-syntax-json-strings": "^7.8.3", + "@babel/plugin-syntax-logical-assignment-operators": "^7.8.3", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3", + "@babel/plugin-syntax-numeric-separator": "^7.8.3", + "@babel/plugin-syntax-object-rest-spread": "^7.8.3", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.3", + "@babel/plugin-syntax-optional-chaining": "^7.8.3", + "@babel/plugin-syntax-top-level-await": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/babel-preset-jest": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/babel-preset-jest/-/babel-preset-jest-29.6.3.tgz", + "integrity": "sha512-0B3bhxR6snWXJZtR/RliHTDPRgn1sNHOR0yVtq/IiQFyuOVjFS+wuio/R4gSNkyYmKmJB4wGZv2NZanmKmTnNA==", + "dev": true, + "dependencies": { + "babel-plugin-jest-hoist": "^29.6.3", + "babel-preset-current-node-syntax": "^1.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.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==", + "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==", + "dev": true, + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.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==", + "dev": true, + "dependencies": { + "fill-range": "^7.1.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/browserslist": { + "version": "4.23.1", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.23.1.tgz", + "integrity": "sha512-TUfofFo/KsK/bWZ9TWQ5O26tsWW4Uhmt8IYklbnUa70udB6P2wA7w7o4PY4muaEPBQaAX+CEnmmIA41NVHtPVw==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "dependencies": { + "caniuse-lite": "^1.0.30001629", + "electron-to-chromium": "^1.4.796", + "node-releases": "^2.0.14", + "update-browserslist-db": "^1.0.16" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/bs-logger": { + "version": "0.2.6", + "resolved": "https://registry.npmjs.org/bs-logger/-/bs-logger-0.2.6.tgz", + "integrity": "sha512-pd8DCoxmbgc7hyPKOvxtqNcjYoOsABPQdcCUjGp3d42VR2CX1ORhk2A87oqqu5R1kk+76nsxZupkmyd+MVtCog==", + "dev": true, + "dependencies": { + "fast-json-stable-stringify": "2.x" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/bser": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/bser/-/bser-2.1.1.tgz", + "integrity": "sha512-gQxTNE/GAfIIrmHLUE3oJyp5FO6HRBfhjnw4/wMmA63ZGDJnWBmgY/lyQBpnDUkGmAhbSe39tx2d/iTOAfglwQ==", + "dev": true, + "dependencies": { + "node-int64": "^0.4.0" + } + }, + "node_modules/buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", + "dev": true + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/camelcase": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", + "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001636", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001636.tgz", + "integrity": "sha512-bMg2vmr8XBsbL6Lr0UHXy/21m84FTxDLWn2FSqMd5PrlbMxwJlQnC2YWYxVgp66PZE+BBNF2jYQUBKCo1FDeZg==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ] + }, + "node_modules/chalk": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-5.3.0.tgz", + "integrity": "sha512-dLitG79d+GV1Nb/VYcCDFivJeK1hiukt9QjRNVOsUtTy1rR1YJsmpGGTZ3qJos+uw7WmWF4wUwBd9jxjocFC2w==", + "engines": { + "node": "^12.17.0 || ^14.13 || >=16.0.0" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/char-regex": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/char-regex/-/char-regex-1.0.2.tgz", + "integrity": "sha512-kWWXztvZ5SBQV+eRgKFeh8q5sLuZY2+8WUIzlxWVTg+oGwY14qylx1KbKzHd8P6ZYkAg0xyIDU9JMHhyJMZ1jw==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/character-entities": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/character-entities/-/character-entities-2.0.2.tgz", + "integrity": "sha512-shx7oQ0Awen/BRIdkjkvz54PnEEI/EjwXDSIZp86/KKdbafHh1Df/RYGBhn4hbe2+uKC9FnT5UCEdyPz3ai9hQ==", + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/ci-info": { + "version": "3.9.0", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.9.0.tgz", + "integrity": "sha512-NIxF55hv4nSqQswkAeiOi1r83xy8JldOFDTWiug55KBu9Jnblncd2U6ViHmYgHf01TPZS77NJBhBMKdWj9HQMQ==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/sibiraj-s" + } + ], + "engines": { + "node": ">=8" + } + }, + "node_modules/cjs-module-lexer": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-1.3.1.tgz", + "integrity": "sha512-a3KdPAANPbNE4ZUv9h6LckSl9zLsYOP4MBmhIPkRaeyybt+r4UghLvq+xw/YwUcC1gqylCkL4rdVs3Lwupjm4Q==", + "dev": true + }, + "node_modules/cjson": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/cjson/-/cjson-0.3.0.tgz", + "integrity": "sha512-bBRQcCIHzI1IVH59fR0bwGrFmi3Btb/JNwM/n401i1DnYgWndpsUBiQRAddLflkZage20A2d25OAWZZk0vBRlA==", + "dependencies": { + "jsonlint": "1.6.0" + }, + "engines": { + "node": ">= 0.3.0" + } + }, + "node_modules/clear": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/clear/-/clear-0.1.0.tgz", + "integrity": "sha512-qMjRnoL+JDPJHeLePZJuao6+8orzHMGP04A8CdwCNsKhRbOnKRjefxONR7bwILT3MHecxKBjHkKL/tkZ8r4Uzw==", + "engines": { + "node": "*" + } + }, + "node_modules/cliui": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", + "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", + "dev": true, + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.1", + "wrap-ansi": "^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/cliui/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/co": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", + "integrity": "sha512-QVb0dM5HvG+uaxitm8wONl7jltx8dqhfU33DcqtOZcLSVIKSDDLDi7+0LbAKiyI8hD9u42m2YxXSkMGWThaecQ==", + "dev": true, + "engines": { + "iojs": ">= 1.0.0", + "node": ">= 0.12.0" + } + }, + "node_modules/collect-v8-coverage": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.2.tgz", + "integrity": "sha512-lHl4d5/ONEbLlJvaJNtsF/Lz+WvB07u2ycqTYbdrq7UypDXailES4valYb2eWiJFxZlVmpGekfqoxQhzyFdT4Q==", + "dev": true + }, + "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/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/colors": { + "version": "0.5.1", + "resolved": "https://registry.npmjs.org/colors/-/colors-0.5.1.tgz", + "integrity": "sha512-XjsuUwpDeY98+yz959OlUK6m7mLBM+1MEG5oaenfuQnNnrQk1WvtcvFgN3FNDP3f2NmZ211t0mNEfSEN1h0eIg==", + "engines": { + "node": ">=0.1.90" + } + }, + "node_modules/commander": { + "version": "11.1.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-11.1.0.tgz", + "integrity": "sha512-yPVavfyCcRhmorC7rWlkHn15b4wDVgVmBA7kV4QVBsF7kv/9TKJAbAXVTxvTnwP8HHKjRCJDClKbciiYS7p0DQ==", + "engines": { + "node": ">=16" + } + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true + }, + "node_modules/cose-base": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/cose-base/-/cose-base-1.0.3.tgz", + "integrity": "sha512-s9whTXInMSgAp/NVXVNuVxVKzGH2qck3aQlVHxDCdAEPgtMKwc4Wq6/QKhgdEdgbLSi9rBTAcPoRa6JpiG4ksg==", + "dependencies": { + "layout-base": "^1.0.0" + } + }, + "node_modules/create-jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/create-jest/-/create-jest-29.7.0.tgz", + "integrity": "sha512-Adz2bdH0Vq3F53KEMJOoftQFutWCukm6J24wbPWRO4k1kMY7gS7ds/uoJkNuV8wDCtWWnuwGcJwpWcih+zEW1Q==", + "dev": true, + "dependencies": { + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "exit": "^0.1.2", + "graceful-fs": "^4.2.9", + "jest-config": "^29.7.0", + "jest-util": "^29.7.0", + "prompts": "^2.0.1" + }, + "bin": { + "create-jest": "bin/create-jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/create-jest/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/create-require": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz", + "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==", + "dev": true + }, + "node_modules/cytoscape": { + "version": "3.29.2", + "resolved": "https://registry.npmjs.org/cytoscape/-/cytoscape-3.29.2.tgz", + "integrity": "sha512-2G1ycU28Nh7OHT9rkXRLpCDP30MKH1dXJORZuBhtEhEW7pKwgPi77ImqlCWinouyE1PNepIOGZBOrE84DG7LyQ==", + "engines": { + "node": ">=0.10" + } + }, + "node_modules/cytoscape-cose-bilkent": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/cytoscape-cose-bilkent/-/cytoscape-cose-bilkent-4.1.0.tgz", + "integrity": "sha512-wgQlVIUJF13Quxiv5e1gstZ08rnZj2XaLHGoFMYXz7SkNfCDOOteKBE6SYRfA9WxxI/iBc3ajfDoc6hb/MRAHQ==", + "dependencies": { + "cose-base": "^1.0.0" + }, + "peerDependencies": { + "cytoscape": "^3.2.0" + } + }, + "node_modules/d3": { + "version": "7.9.0", + "resolved": "https://registry.npmjs.org/d3/-/d3-7.9.0.tgz", + "integrity": "sha512-e1U46jVP+w7Iut8Jt8ri1YsPOvFpg46k+K8TpCb0P+zjCkjkPnV7WzfDJzMHy1LnA+wj5pLT1wjO901gLXeEhA==", + "dependencies": { + "d3-array": "3", + "d3-axis": "3", + "d3-brush": "3", + "d3-chord": "3", + "d3-color": "3", + "d3-contour": "4", + "d3-delaunay": "6", + "d3-dispatch": "3", + "d3-drag": "3", + "d3-dsv": "3", + "d3-ease": "3", + "d3-fetch": "3", + "d3-force": "3", + "d3-format": "3", + "d3-geo": "3", + "d3-hierarchy": "3", + "d3-interpolate": "3", + "d3-path": "3", + "d3-polygon": "3", + "d3-quadtree": "3", + "d3-random": "3", + "d3-scale": "4", + "d3-scale-chromatic": "3", + "d3-selection": "3", + "d3-shape": "3", + "d3-time": "3", + "d3-time-format": "4", + "d3-timer": "3", + "d3-transition": "3", + "d3-zoom": "3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-array": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/d3-array/-/d3-array-3.2.4.tgz", + "integrity": "sha512-tdQAmyA18i4J7wprpYq8ClcxZy3SC31QMeByyCFyRt7BVHdREQZ5lpzoe5mFEYZUWe+oq8HBvk9JjpibyEV4Jg==", + "dependencies": { + "internmap": "1 - 2" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-axis": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/d3-axis/-/d3-axis-3.0.0.tgz", + "integrity": "sha512-IH5tgjV4jE/GhHkRV0HiVYPDtvfjHQlQfJHs0usq7M30XcSBvOotpmH1IgkcXsO/5gEQZD43B//fc7SRT5S+xw==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-brush": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/d3-brush/-/d3-brush-3.0.0.tgz", + "integrity": "sha512-ALnjWlVYkXsVIGlOsuWH1+3udkYFI48Ljihfnh8FZPF2QS9o+PzGLBslO0PjzVoHLZ2KCVgAM8NVkXPJB2aNnQ==", + "dependencies": { + "d3-dispatch": "1 - 3", + "d3-drag": "2 - 3", + "d3-interpolate": "1 - 3", + "d3-selection": "3", + "d3-transition": "3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-chord": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-chord/-/d3-chord-3.0.1.tgz", + "integrity": "sha512-VE5S6TNa+j8msksl7HwjxMHDM2yNK3XCkusIlpX5kwauBfXuyLAtNg9jCp/iHH61tgI4sb6R/EIMWCqEIdjT/g==", + "dependencies": { + "d3-path": "1 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-color": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/d3-color/-/d3-color-3.1.0.tgz", + "integrity": "sha512-zg/chbXyeBtMQ1LbD/WSoW2DpC3I0mpmPdW+ynRTj/x2DAWYrIY7qeZIHidozwV24m4iavr15lNwIwLxRmOxhA==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-contour": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/d3-contour/-/d3-contour-4.0.2.tgz", + "integrity": "sha512-4EzFTRIikzs47RGmdxbeUvLWtGedDUNkTcmzoeyg4sP/dvCexO47AaQL7VKy/gul85TOxw+IBgA8US2xwbToNA==", + "dependencies": { + "d3-array": "^3.2.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-delaunay": { + "version": "6.0.4", + "resolved": "https://registry.npmjs.org/d3-delaunay/-/d3-delaunay-6.0.4.tgz", + "integrity": "sha512-mdjtIZ1XLAM8bm/hx3WwjfHt6Sggek7qH043O8KEjDXN40xi3vx/6pYSVTwLjEgiXQTbvaouWKynLBiUZ6SK6A==", + "dependencies": { + "delaunator": "5" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-dispatch": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-dispatch/-/d3-dispatch-3.0.1.tgz", + "integrity": "sha512-rzUyPU/S7rwUflMyLc1ETDeBj0NRuHKKAcvukozwhshr6g6c5d8zh4c2gQjY2bZ0dXeGLWc1PF174P2tVvKhfg==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-drag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/d3-drag/-/d3-drag-3.0.0.tgz", + "integrity": "sha512-pWbUJLdETVA8lQNJecMxoXfH6x+mO2UQo8rSmZ+QqxcbyA3hfeprFgIT//HW2nlHChWeIIMwS2Fq+gEARkhTkg==", + "dependencies": { + "d3-dispatch": "1 - 3", + "d3-selection": "3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-dsv": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-dsv/-/d3-dsv-3.0.1.tgz", + "integrity": "sha512-UG6OvdI5afDIFP9w4G0mNq50dSOsXHJaRE8arAS5o9ApWnIElp8GZw1Dun8vP8OyHOZ/QJUKUJwxiiCCnUwm+Q==", + "dependencies": { + "commander": "7", + "iconv-lite": "0.6", + "rw": "1" + }, + "bin": { + "csv2json": "bin/dsv2json.js", + "csv2tsv": "bin/dsv2dsv.js", + "dsv2dsv": "bin/dsv2dsv.js", + "dsv2json": "bin/dsv2json.js", + "json2csv": "bin/json2dsv.js", + "json2dsv": "bin/json2dsv.js", + "json2tsv": "bin/json2dsv.js", + "tsv2csv": "bin/dsv2dsv.js", + "tsv2json": "bin/dsv2json.js" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-dsv/node_modules/commander": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-7.2.0.tgz", + "integrity": "sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==", + "engines": { + "node": ">= 10" + } + }, + "node_modules/d3-ease": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-ease/-/d3-ease-3.0.1.tgz", + "integrity": "sha512-wR/XK3D3XcLIZwpbvQwQ5fK+8Ykds1ip7A2Txe0yxncXSdq1L9skcG7blcedkOX+ZcgxGAmLX1FrRGbADwzi0w==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-fetch": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-fetch/-/d3-fetch-3.0.1.tgz", + "integrity": "sha512-kpkQIM20n3oLVBKGg6oHrUchHM3xODkTzjMoj7aWQFq5QEM+R6E4WkzT5+tojDY7yjez8KgCBRoj4aEr99Fdqw==", + "dependencies": { + "d3-dsv": "1 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-force": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/d3-force/-/d3-force-3.0.0.tgz", + "integrity": "sha512-zxV/SsA+U4yte8051P4ECydjD/S+qeYtnaIyAs9tgHCqfguma/aAQDjo85A9Z6EKhBirHRJHXIgJUlffT4wdLg==", + "dependencies": { + "d3-dispatch": "1 - 3", + "d3-quadtree": "1 - 3", + "d3-timer": "1 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-format": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/d3-format/-/d3-format-3.1.0.tgz", + "integrity": "sha512-YyUI6AEuY/Wpt8KWLgZHsIU86atmikuoOmCfommt0LYHiQSPjvX2AcFc38PX0CBpr2RCyZhjex+NS/LPOv6YqA==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-geo": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/d3-geo/-/d3-geo-3.1.1.tgz", + "integrity": "sha512-637ln3gXKXOwhalDzinUgY83KzNWZRKbYubaG+fGVuc/dxO64RRljtCTnf5ecMyE1RIdtqpkVcq0IbtU2S8j2Q==", + "dependencies": { + "d3-array": "2.5.0 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-hierarchy": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/d3-hierarchy/-/d3-hierarchy-3.1.2.tgz", + "integrity": "sha512-FX/9frcub54beBdugHjDCdikxThEqjnR93Qt7PvQTOHxyiNCAlvMrHhclk3cD5VeAaq9fxmfRp+CnWw9rEMBuA==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-interpolate": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-interpolate/-/d3-interpolate-3.0.1.tgz", + "integrity": "sha512-3bYs1rOD33uo8aqJfKP3JWPAibgw8Zm2+L9vBKEHJ2Rg+viTR7o5Mmv5mZcieN+FRYaAOWX5SJATX6k1PWz72g==", + "dependencies": { + "d3-color": "1 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-path": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/d3-path/-/d3-path-3.1.0.tgz", + "integrity": "sha512-p3KP5HCf/bvjBSSKuXid6Zqijx7wIfNW+J/maPs+iwR35at5JCbLUT0LzF1cnjbCHWhqzQTIN2Jpe8pRebIEFQ==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-polygon": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-polygon/-/d3-polygon-3.0.1.tgz", + "integrity": "sha512-3vbA7vXYwfe1SYhED++fPUQlWSYTTGmFmQiany/gdbiWgU/iEyQzyymwL9SkJjFFuCS4902BSzewVGsHHmHtXg==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-quadtree": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-quadtree/-/d3-quadtree-3.0.1.tgz", + "integrity": "sha512-04xDrxQTDTCFwP5H6hRhsRcb9xxv2RzkcsygFzmkSIOJy3PeRJP7sNk3VRIbKXcog561P9oU0/rVH6vDROAgUw==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-random": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-random/-/d3-random-3.0.1.tgz", + "integrity": "sha512-FXMe9GfxTxqd5D6jFsQ+DJ8BJS4E/fT5mqqdjovykEB2oFbTMDVdg1MGFxfQW+FBOGoB++k8swBrgwSHT1cUXQ==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-sankey": { + "version": "0.12.3", + "resolved": "https://registry.npmjs.org/d3-sankey/-/d3-sankey-0.12.3.tgz", + "integrity": "sha512-nQhsBRmM19Ax5xEIPLMY9ZmJ/cDvd1BG3UVvt5h3WRxKg5zGRbvnteTyWAbzeSvlh3tW7ZEmq4VwR5mB3tutmQ==", + "dependencies": { + "d3-array": "1 - 2", + "d3-shape": "^1.2.0" + } + }, + "node_modules/d3-sankey/node_modules/d3-array": { + "version": "2.12.1", + "resolved": "https://registry.npmjs.org/d3-array/-/d3-array-2.12.1.tgz", + "integrity": "sha512-B0ErZK/66mHtEsR1TkPEEkwdy+WDesimkM5gpZr5Dsg54BiTA5RXtYW5qTLIAcekaS9xfZrzBLF/OAkB3Qn1YQ==", + "dependencies": { + "internmap": "^1.0.0" + } + }, + "node_modules/d3-sankey/node_modules/d3-path": { + "version": "1.0.9", + "resolved": "https://registry.npmjs.org/d3-path/-/d3-path-1.0.9.tgz", + "integrity": "sha512-VLaYcn81dtHVTjEHd8B+pbe9yHWpXKZUC87PzoFmsFrJqgFwDe/qxfp5MlfsfM1V5E/iVt0MmEbWQ7FVIXh/bg==" + }, + "node_modules/d3-sankey/node_modules/d3-shape": { + "version": "1.3.7", + "resolved": "https://registry.npmjs.org/d3-shape/-/d3-shape-1.3.7.tgz", + "integrity": "sha512-EUkvKjqPFUAZyOlhY5gzCxCeI0Aep04LwIRpsZ/mLFelJiUfnK56jo5JMDSE7yyP2kLSb6LtF+S5chMk7uqPqw==", + "dependencies": { + "d3-path": "1" + } + }, + "node_modules/d3-sankey/node_modules/internmap": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/internmap/-/internmap-1.0.1.tgz", + "integrity": "sha512-lDB5YccMydFBtasVtxnZ3MRBHuaoE8GKsppq+EchKL2U4nK/DmEpPHNH8MZe5HkMtpSiTSOZwfN0tzYjO/lJEw==" + }, + "node_modules/d3-scale": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/d3-scale/-/d3-scale-4.0.2.tgz", + "integrity": "sha512-GZW464g1SH7ag3Y7hXjf8RoUuAFIqklOAq3MRl4OaWabTFJY9PN/E1YklhXLh+OQ3fM9yS2nOkCoS+WLZ6kvxQ==", + "dependencies": { + "d3-array": "2.10.0 - 3", + "d3-format": "1 - 3", + "d3-interpolate": "1.2.0 - 3", + "d3-time": "2.1.1 - 3", + "d3-time-format": "2 - 4" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-scale-chromatic": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/d3-scale-chromatic/-/d3-scale-chromatic-3.1.0.tgz", + "integrity": "sha512-A3s5PWiZ9YCXFye1o246KoscMWqf8BsD9eRiJ3He7C9OBaxKhAd5TFCdEx/7VbKtxxTsu//1mMJFrEt572cEyQ==", + "dependencies": { + "d3-color": "1 - 3", + "d3-interpolate": "1 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-selection": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/d3-selection/-/d3-selection-3.0.0.tgz", + "integrity": "sha512-fmTRWbNMmsmWq6xJV8D19U/gw/bwrHfNXxrIN+HfZgnzqTHp9jOmKMhsTUjXOJnZOdZY9Q28y4yebKzqDKlxlQ==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-shape": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/d3-shape/-/d3-shape-3.2.0.tgz", + "integrity": "sha512-SaLBuwGm3MOViRq2ABk3eLoxwZELpH6zhl3FbAoJ7Vm1gofKx6El1Ib5z23NUEhF9AsGl7y+dzLe5Cw2AArGTA==", + "dependencies": { + "d3-path": "^3.1.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-time": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/d3-time/-/d3-time-3.1.0.tgz", + "integrity": "sha512-VqKjzBLejbSMT4IgbmVgDjpkYrNWUYJnbCGo874u7MMKIWsILRX+OpX/gTk8MqjpT1A/c6HY2dCA77ZN0lkQ2Q==", + "dependencies": { + "d3-array": "2 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-time-format": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/d3-time-format/-/d3-time-format-4.1.0.tgz", + "integrity": "sha512-dJxPBlzC7NugB2PDLwo9Q8JiTR3M3e4/XANkreKSUxF8vvXKqm1Yfq4Q5dl8budlunRVlUUaDUgFt7eA8D6NLg==", + "dependencies": { + "d3-time": "1 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-timer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-timer/-/d3-timer-3.0.1.tgz", + "integrity": "sha512-ndfJ/JxxMd3nw31uyKoY2naivF+r29V+Lc0svZxe1JvvIRmi8hUsrMvdOwgS1o6uBHmiz91geQ0ylPP0aj1VUA==", + "engines": { + "node": ">=12" + } + }, + "node_modules/d3-transition": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/d3-transition/-/d3-transition-3.0.1.tgz", + "integrity": "sha512-ApKvfjsSR6tg06xrL434C0WydLr7JewBB3V+/39RMHsaXTOG0zmt/OAXeng5M5LBm0ojmxJrpomQVZ1aPvBL4w==", + "dependencies": { + "d3-color": "1 - 3", + "d3-dispatch": "1 - 3", + "d3-ease": "1 - 3", + "d3-interpolate": "1 - 3", + "d3-timer": "1 - 3" + }, + "engines": { + "node": ">=12" + }, + "peerDependencies": { + "d3-selection": "2 - 3" + } + }, + "node_modules/d3-zoom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/d3-zoom/-/d3-zoom-3.0.0.tgz", + "integrity": "sha512-b8AmV3kfQaqWAuacbPuNbL6vahnOJflOhexLzMMNLga62+/nh0JzvJ0aO/5a5MVgUFGS7Hu1P9P03o3fJkDCyw==", + "dependencies": { + "d3-dispatch": "1 - 3", + "d3-drag": "2 - 3", + "d3-interpolate": "1 - 3", + "d3-selection": "2 - 3", + "d3-transition": "2 - 3" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/dagre-d3-es": { + "version": "7.0.10", + "resolved": "https://registry.npmjs.org/dagre-d3-es/-/dagre-d3-es-7.0.10.tgz", + "integrity": "sha512-qTCQmEhcynucuaZgY5/+ti3X/rnszKZhEQH/ZdWdtP1tA/y3VoHJzcVrO9pjjJCNpigfscAtoUB5ONcd2wNn0A==", + "dependencies": { + "d3": "^7.8.2", + "lodash-es": "^4.17.21" + } + }, + "node_modules/dayjs": { + "version": "1.11.11", + "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.11.11.tgz", + "integrity": "sha512-okzr3f11N6WuqYtZSvm+F776mB41wRZMhKP+hc34YdW+KmtYYK9iqvHSwo2k9FEH3fhGXvOPV6yz2IcSrfRUDg==" + }, + "node_modules/debug": { + "version": "4.3.5", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.5.tgz", + "integrity": "sha512-pt0bNEmneDIvdL1Xsd9oDQ/wrQRkXDT4AUWlNZNPKvW5x/jyO9VFXkJUP07vQ2upmw5PlaITaPKc31jK13V+jg==", + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/decode-named-character-reference": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/decode-named-character-reference/-/decode-named-character-reference-1.0.2.tgz", + "integrity": "sha512-O8x12RzrUF8xyVcY0KJowWsmaJxQbmy0/EtnNtHRpsOcT7dFk5W598coHqBVpmWo1oQQfsCqfCmkZN5DJrZVdg==", + "dependencies": { + "character-entities": "^2.0.0" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/dedent": { + "version": "1.5.3", + "resolved": "https://registry.npmjs.org/dedent/-/dedent-1.5.3.tgz", + "integrity": "sha512-NHQtfOOW68WD8lgypbLA5oT+Bt0xXJhiYvoR6SmmNXZfpzOGXwdKWmcwG8N7PwVVWV3eF/68nmD9BaJSsTBhyQ==", + "dev": true, + "peerDependencies": { + "babel-plugin-macros": "^3.1.0" + }, + "peerDependenciesMeta": { + "babel-plugin-macros": { + "optional": true + } + } + }, + "node_modules/deepmerge": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.3.1.tgz", + "integrity": "sha512-3sUqbMEc77XqpdNO7FRyRog+eW3ph+GYCbj+rK+uYyRMuwsVy0rMiVtPn+QJlKFvWP/1PYpapqYn0Me2knFn+A==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/delaunator": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/delaunator/-/delaunator-5.0.1.tgz", + "integrity": "sha512-8nvh+XBe96aCESrGOqMp/84b13H9cdKbG5P2ejQCh4d4sK9RL4371qou9drQjMhvnPmhWl5hnmqbEE0fXr9Xnw==", + "dependencies": { + "robust-predicates": "^3.0.2" + } + }, + "node_modules/dequal": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", + "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==", + "engines": { + "node": ">=6" + } + }, + "node_modules/detect-newline": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz", + "integrity": "sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/diff": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/diff/-/diff-5.2.0.tgz", + "integrity": "sha512-uIFDxqpRZGZ6ThOk84hEfqWoHx2devRFvpTZcTHur85vImfaxUbTW9Ryh4CpCuDnToOP1CEtXKIgytHBPVff5A==", + "engines": { + "node": ">=0.3.1" + } + }, + "node_modules/dompurify": { + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/dompurify/-/dompurify-3.1.5.tgz", + "integrity": "sha512-lwG+n5h8QNpxtyrJW/gJWckL+1/DQiYMX8f7t8Z2AZTPw1esVrqjI63i7Zc2Gz0aKzLVMYC1V1PL/ky+aY/NgA==" + }, + "node_modules/ebnf-parser": { + "version": "0.1.10", + "resolved": "https://registry.npmjs.org/ebnf-parser/-/ebnf-parser-0.1.10.tgz", + "integrity": "sha512-urvSxVQ6XJcoTpc+/x2pWhhuOX4aljCNQpwzw+ifZvV1andZkAmiJc3Rq1oGEAQmcjiLceyMXOy1l8ms8qs2fQ==" + }, + "node_modules/electron-to-chromium": { + "version": "1.4.810", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.810.tgz", + "integrity": "sha512-Kaxhu4T7SJGpRQx99tq216gCq2nMxJo+uuT6uzz9l8TVN2stL7M06MIIXAtr9jsrLs2Glflgf2vMQRepxawOdQ==", + "dev": true + }, + "node_modules/elkjs": { + "version": "0.9.3", + "resolved": "https://registry.npmjs.org/elkjs/-/elkjs-0.9.3.tgz", + "integrity": "sha512-f/ZeWvW/BCXbhGEf1Ujp29EASo/lk1FDnETgNKwJrsVvGZhUWCZyg3xLJjAsxfOmt8KjswHmI5EwCQcPMpOYhQ==" + }, + "node_modules/emittery": { + "version": "0.13.1", + "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.13.1.tgz", + "integrity": "sha512-DeWwawk6r5yR9jFgnDKYt4sLS0LmHJJi3ZOnb5/JdbYwj3nW+FxQnHIjhBKz8YLC7oRNPVM9NQ47I3CVx34eqQ==", + "dev": true, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sindresorhus/emittery?sponsor=1" + } + }, + "node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true + }, + "node_modules/error-ex": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", + "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", + "dev": true, + "dependencies": { + "is-arrayish": "^0.2.1" + } + }, + "node_modules/escalade": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.2.tgz", + "integrity": "sha512-ErCHMCae19vR8vQGe50xIsVomy19rg6gFu3+r3jkEO46suLMWBksvVyoGgQV+jOfl84ZSOSlmv6Gxa89PmTGmA==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/escodegen": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.3.3.tgz", + "integrity": "sha512-z9FWgKc48wjMlpzF5ymKS1AF8OIgnKLp9VyN7KbdtyrP/9lndwUFqCtMm+TAJmJf7KJFFYc4cFJfVTTGkKEwsA==", + "dependencies": { + "esprima": "~1.1.1", + "estraverse": "~1.5.0", + "esutils": "~1.0.0" + }, + "bin": { + "escodegen": "bin/escodegen.js", + "esgenerate": "bin/esgenerate.js" + }, + "engines": { + "node": ">=0.10.0" + }, + "optionalDependencies": { + "source-map": "~0.1.33" + } + }, + "node_modules/esprima": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-1.1.1.tgz", + "integrity": "sha512-qxxB994/7NtERxgXdFgLHIs9M6bhLXc6qtUmWZ3L8+gTQ9qaoyki2887P2IqAYsoENyr8SUbTutStDniOHSDHg==", + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/estraverse": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-1.5.1.tgz", + "integrity": "sha512-FpCjJDfmo3vsc/1zKSeqR5k42tcIhxFIlvq+h9j0fO2q/h2uLKyweq7rYJ+0CoVvrGQOxIS5wyBrW/+vF58BUQ==", + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/esutils": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-1.0.0.tgz", + "integrity": "sha512-x/iYH53X3quDwfHRz4y8rn4XcEwwCJeWsul9pF1zldMbGtgOtMNBEOuYWwB1EQlK2LRa1fev3YAgym/RElp5Cg==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/execa": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/execa/-/execa-5.1.1.tgz", + "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==", + "dev": true, + "dependencies": { + "cross-spawn": "^7.0.3", + "get-stream": "^6.0.0", + "human-signals": "^2.1.0", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.1", + "onetime": "^5.1.2", + "signal-exit": "^3.0.3", + "strip-final-newline": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sindresorhus/execa?sponsor=1" + } + }, + "node_modules/execa/node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/execa/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/execa/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/execa/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/execa/node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/exit": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", + "integrity": "sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/expect": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/expect/-/expect-29.7.0.tgz", + "integrity": "sha512-2Zks0hf1VLFYI1kbh0I5jP3KHHyCHpkfyHBzsSXRFgl/Bg9mWYfMW8oD+PdMPlEwy5HNsR9JutYy6pMeOh61nw==", + "dev": true, + "dependencies": { + "@jest/expect-utils": "^29.7.0", + "jest-get-type": "^29.6.3", + "jest-matcher-utils": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/expect/node_modules/jest-get-type": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", + "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "dev": true, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "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/fb-watchman": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/fb-watchman/-/fb-watchman-2.0.2.tgz", + "integrity": "sha512-p5161BqbuCaSnB8jIbzQHOlpgsPmK5rJVDfDKO91Axs5NC1uu3HRQm6wt9cd9/+GtQQIO53JdGXXoyDpTAsgYA==", + "dev": true, + "dependencies": { + "bser": "2.1.1" + } + }, + "node_modules/figlet": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/figlet/-/figlet-1.7.0.tgz", + "integrity": "sha512-gO8l3wvqo0V7wEFLXPbkX83b7MVjRrk1oRLfYlZXol8nEpb/ON9pcKLI4qpBv5YtOTfrINtqb7b40iYY2FTWFg==", + "bin": { + "figlet": "bin/index.js" + }, + "engines": { + "node": ">= 0.4.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, + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dev": true, + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dev": true + }, + "node_modules/fsevents": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", + "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==", + "dev": true, + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "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, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/get-caller-file": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", + "dev": true, + "engines": { + "node": "6.* || 8.* || >= 10.*" + } + }, + "node_modules/get-package-type": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/get-package-type/-/get-package-type-0.1.0.tgz", + "integrity": "sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==", + "dev": true, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/get-stream": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-6.0.1.tgz", + "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/glob": { + "version": "7.2.3", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "deprecated": "Glob versions prior to v9 are no longer supported", + "dev": true, + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.1.1", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/graceful-fs": { + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", + "dev": true + }, + "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/hasown": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", + "dev": true, + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/html-escaper": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", + "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", + "dev": true + }, + "node_modules/human-signals": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz", + "integrity": "sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==", + "dev": true, + "engines": { + "node": ">=10.17.0" + } + }, + "node_modules/iconv-lite": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/import-local": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.1.0.tgz", + "integrity": "sha512-ASB07uLtnDs1o6EHjKpX34BKYDSqnFerfTOJL2HvMqF70LnxpjkzDB8J44oT9pu4AMPkQwf8jl6szgvNd2tRIg==", + "dev": true, + "dependencies": { + "pkg-dir": "^4.2.0", + "resolve-cwd": "^3.0.0" + }, + "bin": { + "import-local-fixture": "fixtures/cli.js" + }, + "engines": { + "node": ">=8" + }, + "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": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true, + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "deprecated": "This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.", + "dev": true, + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha512-x00IRNXNy63jwGkJmzPigoySHbaqpNuzKbBOmzK+g2OdZpQ9w+sxCN+VSB3ja7IAge2OP2qpfxTjeNcyjmW1uw==" + }, + "node_modules/internmap": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/internmap/-/internmap-2.0.3.tgz", + "integrity": "sha512-5Hh7Y1wQbvY5ooGgPbDaL5iYLAPzMTUrjMulskHLH6wnv/A+1q5rgEaiuqEjB+oxGXIVZs1FF+R/KPN3ZSQYYg==", + "engines": { + "node": ">=12" + } + }, + "node_modules/is-arrayish": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", + "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", + "dev": true + }, + "node_modules/is-core-module": { + "version": "2.14.0", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.14.0.tgz", + "integrity": "sha512-a5dFJih5ZLYlRtDc0dZWP7RiKr6xIKzmn/oAYCDvdLThadVgyJwlaoQPmRtMSpz+rk0OGAgIu+TcM9HUF0fk1A==", + "dev": true, + "dependencies": { + "hasown": "^2.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/is-generator-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-generator-fn/-/is-generator-fn-2.1.0.tgz", + "integrity": "sha512-cTIB4yPYL/Grw0EaSzASzg6bBy9gqCofvWN8okThAYIxKJZC+udlRAmGbM0XLeniEJSs8uEgHPGuHSe1XsOLSQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "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, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/is-stream": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", + "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", + "dev": true, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "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/istanbul-lib-coverage": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.2.2.tgz", + "integrity": "sha512-O8dpsF+r0WV/8MNRKfnmrtCWhuKjxrq2w+jpzBL5UZKTi2LeVWnWOmWRxFlesJONmc+wLAGvKQZEOanko0LFTg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-instrument": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-6.0.2.tgz", + "integrity": "sha512-1WUsZ9R1lA0HtBSohTkm39WTPlNKSJ5iFk7UwqXkBLoHQT+hfqPsfsTDVuZdKGaBwn7din9bS7SsnoAr943hvw==", + "dev": true, + "dependencies": { + "@babel/core": "^7.23.9", + "@babel/parser": "^7.23.9", + "@istanbuljs/schema": "^0.1.3", + "istanbul-lib-coverage": "^3.2.0", + "semver": "^7.5.4" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-report": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.1.tgz", + "integrity": "sha512-GCfE1mtsHGOELCU8e/Z7YWzpmybrx/+dSTfLrvY8qRmaY6zXTKWn6WQIjaAFw069icm6GVMNkgu0NzI4iPZUNw==", + "dev": true, + "dependencies": { + "istanbul-lib-coverage": "^3.0.0", + "make-dir": "^4.0.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-source-maps": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.1.tgz", + "integrity": "sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw==", + "dev": true, + "dependencies": { + "debug": "^4.1.1", + "istanbul-lib-coverage": "^3.0.0", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/istanbul-lib-source-maps/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/istanbul-reports": { + "version": "3.1.7", + "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.1.7.tgz", + "integrity": "sha512-BewmUXImeuRk2YY0PVbxgKAysvhRPUQE0h5QRM++nVWyubKGV0l8qQ5op8+B2DOmwSe63Jivj0BjkPQVf8fP5g==", + "dev": true, + "dependencies": { + "html-escaper": "^2.0.0", + "istanbul-lib-report": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest/-/jest-29.7.0.tgz", + "integrity": "sha512-NIy3oAFp9shda19hy4HK0HRTWKtPJmGdnvywu01nOqNC2vZg+Z+fvJDxpMQA88eb2I9EcafcdjYgsDthnYTvGw==", + "dev": true, + "dependencies": { + "@jest/core": "^29.7.0", + "@jest/types": "^29.6.3", + "import-local": "^3.0.2", + "jest-cli": "^29.7.0" + }, + "bin": { + "jest": "bin/jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/jest-changed-files": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-changed-files/-/jest-changed-files-29.7.0.tgz", + "integrity": "sha512-fEArFiwf1BpQ+4bXSprcDc3/x4HSzL4al2tozwVpDFpsxALjLYdyiIK4e5Vz66GQJIbXJ82+35PtysofptNX2w==", + "dev": true, + "dependencies": { + "execa": "^5.0.0", + "jest-util": "^29.7.0", + "p-limit": "^3.1.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-circus": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-circus/-/jest-circus-29.7.0.tgz", + "integrity": "sha512-3E1nCMgipcTkCocFwM90XXQab9bS+GMsjdpmPrlelaxwD93Ad8iVEjX/vvHPdLPnFf+L40u+5+iutRdA1N9myw==", + "dev": true, + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/expect": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "co": "^4.6.0", + "dedent": "^1.0.0", + "is-generator-fn": "^2.0.0", + "jest-each": "^29.7.0", + "jest-matcher-utils": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-runtime": "^29.7.0", + "jest-snapshot": "^29.7.0", + "jest-util": "^29.7.0", + "p-limit": "^3.1.0", + "pretty-format": "^29.7.0", + "pure-rand": "^6.0.0", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-circus/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/jest-circus/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-circus/node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-circus/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true + }, + "node_modules/jest-cli": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-cli/-/jest-cli-29.7.0.tgz", + "integrity": "sha512-OVVobw2IubN/GSYsxETi+gOe7Ka59EFMR/twOU3Jb2GnKKeMGJB5SGUUrEz3SFVmJASUdZUzy83sLNNQ2gZslg==", + "dev": true, + "dependencies": { + "@jest/core": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "create-jest": "^29.7.0", + "exit": "^0.1.2", + "import-local": "^3.0.2", + "jest-config": "^29.7.0", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "yargs": "^17.3.1" + }, + "bin": { + "jest": "bin/jest.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "node-notifier": "^8.0.1 || ^9.0.0 || ^10.0.0" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/jest-cli/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/jest-config": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-config/-/jest-config-29.7.0.tgz", + "integrity": "sha512-uXbpfeQ7R6TZBqI3/TxCU4q4ttk3u0PJeC+E0zbfSoSjq6bJ7buBPxzQPL0ifrkY4DNu4JUdk0ImlBUYi840eQ==", + "dev": true, + "dependencies": { + "@babel/core": "^7.11.6", + "@jest/test-sequencer": "^29.7.0", + "@jest/types": "^29.6.3", + "babel-jest": "^29.7.0", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "deepmerge": "^4.2.2", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "jest-circus": "^29.7.0", + "jest-environment-node": "^29.7.0", + "jest-get-type": "^29.6.3", + "jest-regex-util": "^29.6.3", + "jest-resolve": "^29.7.0", + "jest-runner": "^29.7.0", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "micromatch": "^4.0.4", + "parse-json": "^5.2.0", + "pretty-format": "^29.7.0", + "slash": "^3.0.0", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "peerDependencies": { + "@types/node": "*", + "ts-node": ">=9.0.0" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "ts-node": { + "optional": true + } + } + }, + "node_modules/jest-config/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/jest-config/node_modules/jest-get-type": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", + "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "dev": true, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-config/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-config/node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-config/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true + }, + "node_modules/jest-docblock": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-29.7.0.tgz", + "integrity": "sha512-q617Auw3A612guyaFgsbFeYpNP5t2aoUNLwBUbc/0kD1R4t9ixDbyFTHd1nok4epoVFpr7PmeWHrhvuV3XaJ4g==", + "dev": true, + "dependencies": { + "detect-newline": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-each": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-each/-/jest-each-29.7.0.tgz", + "integrity": "sha512-gns+Er14+ZrEoC5fhOfYCY1LOHHr0TI+rQUHZS8Ttw2l7gl+80eHc/gFf2Ktkw0+SIACDTeWvpFcv3B04VembQ==", + "dev": true, + "dependencies": { + "@jest/types": "^29.6.3", + "chalk": "^4.0.0", + "jest-get-type": "^29.6.3", + "jest-util": "^29.7.0", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-each/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/jest-each/node_modules/jest-get-type": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", + "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "dev": true, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-each/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-each/node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-each/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true + }, + "node_modules/jest-environment-node": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-environment-node/-/jest-environment-node-29.7.0.tgz", + "integrity": "sha512-DOSwCRqXirTOyheM+4d5YZOrWcdu0LNZ87ewUoywbcb2XR4wKgqiG8vNeYwhjFMbEkfju7wx2GYH0P2gevGvFw==", + "dev": true, + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/fake-timers": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "jest-mock": "^29.7.0", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-haste-map": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-29.7.0.tgz", + "integrity": "sha512-fP8u2pyfqx0K1rGn1R9pyE0/KTn+G7PxktWidOBTqFPLYX0b9ksaMFkhK5vrS3DVun09pckLdlx90QthlW7AmA==", + "dev": true, + "dependencies": { + "@jest/types": "^29.6.3", + "@types/graceful-fs": "^4.1.3", + "@types/node": "*", + "anymatch": "^3.0.3", + "fb-watchman": "^2.0.0", + "graceful-fs": "^4.2.9", + "jest-regex-util": "^29.6.3", + "jest-util": "^29.7.0", + "jest-worker": "^29.7.0", + "micromatch": "^4.0.4", + "walker": "^1.0.8" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + }, + "optionalDependencies": { + "fsevents": "^2.3.2" + } + }, + "node_modules/jest-junit": { + "version": "16.0.0", + "resolved": "https://registry.npmjs.org/jest-junit/-/jest-junit-16.0.0.tgz", + "integrity": "sha512-A94mmw6NfJab4Fg/BlvVOUXzXgF0XIH6EmTgJ5NDPp4xoKq0Kr7sErb+4Xs9nZvu58pJojz5RFGpqnZYJTrRfQ==", + "dev": true, + "dependencies": { + "mkdirp": "^1.0.4", + "strip-ansi": "^6.0.1", + "uuid": "^8.3.2", + "xml": "^1.0.1" + }, + "engines": { + "node": ">=10.12.0" + } + }, + "node_modules/jest-junit/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/jest-junit/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-junit/node_modules/uuid": { + "version": "8.3.2", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz", + "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==", + "dev": true, + "bin": { + "uuid": "dist/bin/uuid" + } + }, + "node_modules/jest-leak-detector": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-leak-detector/-/jest-leak-detector-29.7.0.tgz", + "integrity": "sha512-kYA8IJcSYtST2BY9I+SMC32nDpBT3J2NvWJx8+JCuCdl/CR1I4EKUJROiP8XtCcxqgTTBGJNdbB1A8XRKbTetw==", + "dev": true, + "dependencies": { + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-leak-detector/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-leak-detector/node_modules/jest-get-type": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", + "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "dev": true, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-leak-detector/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-leak-detector/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true + }, + "node_modules/jest-matcher-utils": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-29.7.0.tgz", + "integrity": "sha512-sBkD+Xi9DtcChsI3L3u0+N0opgPYnCRPtGcQYrgXmR+hmt/fYfWAL0xRXYU8eWOdfuLgBe0YCW3AFtnRLagq/g==", + "dev": true, + "dependencies": { + "chalk": "^4.0.0", + "jest-diff": "^29.7.0", + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-matcher-utils/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/jest-matcher-utils/node_modules/diff-sequences": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-29.6.3.tgz", + "integrity": "sha512-EjePK1srD3P08o2j4f0ExnylqRs5B9tJjcp9t1krH2qRi8CCdsYfwe9JgSLurFBWwq4uOlipzfk5fHNvwFKr8Q==", + "dev": true, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-matcher-utils/node_modules/jest-diff": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-29.7.0.tgz", + "integrity": "sha512-LMIgiIrhigmPrs03JHpxUh2yISK3vLFPkAodPeo0+BuF7wA2FoQbkEg1u8gBYBThncu7e1oEDUfIXVuTqLRUjw==", + "dev": true, + "dependencies": { + "chalk": "^4.0.0", + "diff-sequences": "^29.6.3", + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-matcher-utils/node_modules/jest-get-type": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", + "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "dev": true, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-matcher-utils/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-matcher-utils/node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-matcher-utils/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true + }, + "node_modules/jest-message-util": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-29.7.0.tgz", + "integrity": "sha512-GBEV4GRADeP+qtB2+6u61stea8mGcOT4mCtrYISZwfu9/ISHFJ/5zOMXYbpBE9RsS5+Gb63DW4FgmnKJ79Kf6w==", + "dev": true, + "dependencies": { + "@babel/code-frame": "^7.12.13", + "@jest/types": "^29.6.3", + "@types/stack-utils": "^2.0.0", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "micromatch": "^4.0.4", + "pretty-format": "^29.7.0", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-message-util/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/jest-message-util/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-message-util/node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-message-util/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true + }, + "node_modules/jest-mock": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-mock/-/jest-mock-29.7.0.tgz", + "integrity": "sha512-ITOMZn+UkYS4ZFh83xYAOzWStloNzJFO2s8DWrE4lhtGD+AorgnbkiKERe4wQVBydIGPx059g6riW5Btp6Llnw==", + "dev": true, + "dependencies": { + "@jest/types": "^29.6.3", + "@types/node": "*", + "jest-util": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-pnp-resolver": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/jest-pnp-resolver/-/jest-pnp-resolver-1.2.3.tgz", + "integrity": "sha512-+3NpwQEnRoIBtx4fyhblQDPgJI0H1IEIkX7ShLUjPGA7TtUTvI1oiKi3SR4oBR0hQhQR80l4WAe5RrXBwWMA8w==", + "dev": true, + "engines": { + "node": ">=6" + }, + "peerDependencies": { + "jest-resolve": "*" + }, + "peerDependenciesMeta": { + "jest-resolve": { + "optional": true + } + } + }, + "node_modules/jest-regex-util": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-29.6.3.tgz", + "integrity": "sha512-KJJBsRCyyLNWCNBOvZyRDnAIfUiRJ8v+hOBQYGn8gDyF3UegwiP4gwRR3/SDa42g1YbVycTidUF3rKjyLFDWbg==", + "dev": true, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-resolve": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-resolve/-/jest-resolve-29.7.0.tgz", + "integrity": "sha512-IOVhZSrg+UvVAshDSDtHyFCCBUl/Q3AAJv8iZ6ZjnZ74xzvwuzLXid9IIIPgTnY62SJjfuupMKZsZQRsCvxEgA==", + "dev": true, + "dependencies": { + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "jest-pnp-resolver": "^1.2.2", + "jest-util": "^29.7.0", + "jest-validate": "^29.7.0", + "resolve": "^1.20.0", + "resolve.exports": "^2.0.0", + "slash": "^3.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-resolve-dependencies": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-resolve-dependencies/-/jest-resolve-dependencies-29.7.0.tgz", + "integrity": "sha512-un0zD/6qxJ+S0et7WxeI3H5XSe9lTBBR7bOHCHXkKR6luG5mwDDlIzVQ0V5cZCuoTgEdcdwzTghYkTWfubi+nA==", + "dev": true, + "dependencies": { + "jest-regex-util": "^29.6.3", + "jest-snapshot": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-resolve/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/jest-runner": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-runner/-/jest-runner-29.7.0.tgz", + "integrity": "sha512-fsc4N6cPCAahybGBfTRcq5wFR6fpLznMg47sY5aDpsoejOcVYFb07AHuSnR0liMcPTgBsA3ZJL6kFOjPdoNipQ==", + "dev": true, + "dependencies": { + "@jest/console": "^29.7.0", + "@jest/environment": "^29.7.0", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "emittery": "^0.13.1", + "graceful-fs": "^4.2.9", + "jest-docblock": "^29.7.0", + "jest-environment-node": "^29.7.0", + "jest-haste-map": "^29.7.0", + "jest-leak-detector": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-resolve": "^29.7.0", + "jest-runtime": "^29.7.0", + "jest-util": "^29.7.0", + "jest-watcher": "^29.7.0", + "jest-worker": "^29.7.0", + "p-limit": "^3.1.0", + "source-map-support": "0.5.13" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-runner/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/jest-runtime": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-runtime/-/jest-runtime-29.7.0.tgz", + "integrity": "sha512-gUnLjgwdGqW7B4LvOIkbKs9WGbn+QLqRQQ9juC6HndeDiezIwhDP+mhMwHWCEcfQ5RUXa6OPnFF8BJh5xegwwQ==", + "dev": true, + "dependencies": { + "@jest/environment": "^29.7.0", + "@jest/fake-timers": "^29.7.0", + "@jest/globals": "^29.7.0", + "@jest/source-map": "^29.6.3", + "@jest/test-result": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "cjs-module-lexer": "^1.0.0", + "collect-v8-coverage": "^1.0.0", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-mock": "^29.7.0", + "jest-regex-util": "^29.6.3", + "jest-resolve": "^29.7.0", + "jest-snapshot": "^29.7.0", + "jest-util": "^29.7.0", + "slash": "^3.0.0", + "strip-bom": "^4.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-runtime/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/jest-serial-runner": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/jest-serial-runner/-/jest-serial-runner-1.2.1.tgz", + "integrity": "sha512-d59fF+7HdjNvQEL7B4WyFE+f8q5tGzlNUqtOnxTrT1ofun7O6/Lgm/j255BBgCY2fmSue/34M7Xy9+VWRByP0Q==", + "dev": true, + "peerDependencies": { + "jest-runner": "24.x - 29.x" + } + }, + "node_modules/jest-snapshot": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-snapshot/-/jest-snapshot-29.7.0.tgz", + "integrity": "sha512-Rm0BMWtxBcioHr1/OX5YCP8Uov4riHvKPknOGs804Zg9JGZgmIBkbtlxJC/7Z4msKYVbIJtfU+tKb8xlYNfdkw==", + "dev": true, + "dependencies": { + "@babel/core": "^7.11.6", + "@babel/generator": "^7.7.2", + "@babel/plugin-syntax-jsx": "^7.7.2", + "@babel/plugin-syntax-typescript": "^7.7.2", + "@babel/types": "^7.3.3", + "@jest/expect-utils": "^29.7.0", + "@jest/transform": "^29.7.0", + "@jest/types": "^29.6.3", + "babel-preset-current-node-syntax": "^1.0.0", + "chalk": "^4.0.0", + "expect": "^29.7.0", + "graceful-fs": "^4.2.9", + "jest-diff": "^29.7.0", + "jest-get-type": "^29.6.3", + "jest-matcher-utils": "^29.7.0", + "jest-message-util": "^29.7.0", + "jest-util": "^29.7.0", + "natural-compare": "^1.4.0", + "pretty-format": "^29.7.0", + "semver": "^7.5.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-snapshot/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/jest-snapshot/node_modules/diff-sequences": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-29.6.3.tgz", + "integrity": "sha512-EjePK1srD3P08o2j4f0ExnylqRs5B9tJjcp9t1krH2qRi8CCdsYfwe9JgSLurFBWwq4uOlipzfk5fHNvwFKr8Q==", + "dev": true, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-snapshot/node_modules/jest-diff": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-29.7.0.tgz", + "integrity": "sha512-LMIgiIrhigmPrs03JHpxUh2yISK3vLFPkAodPeo0+BuF7wA2FoQbkEg1u8gBYBThncu7e1oEDUfIXVuTqLRUjw==", + "dev": true, + "dependencies": { + "chalk": "^4.0.0", + "diff-sequences": "^29.6.3", + "jest-get-type": "^29.6.3", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-snapshot/node_modules/jest-get-type": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", + "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "dev": true, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-snapshot/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-snapshot/node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-snapshot/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true + }, + "node_modules/jest-util": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-29.7.0.tgz", + "integrity": "sha512-z6EbKajIpqGKU56y5KBUgy1dt1ihhQJgWzUlZHArA/+X2ad7Cb5iF+AK1EWVL/Bo7Rz9uurpqw6SiBCefUbCGA==", + "dev": true, + "dependencies": { + "@jest/types": "^29.6.3", + "@types/node": "*", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "graceful-fs": "^4.2.9", + "picomatch": "^2.2.3" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-util/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/jest-validate": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-validate/-/jest-validate-29.7.0.tgz", + "integrity": "sha512-ZB7wHqaRGVw/9hST/OuFUReG7M8vKeq0/J2egIGLdvjHCmYqGARhzXmtgi+gVeZ5uXFF219aOc3Ls2yLg27tkw==", + "dev": true, + "dependencies": { + "@jest/types": "^29.6.3", + "camelcase": "^6.2.0", + "chalk": "^4.0.0", + "jest-get-type": "^29.6.3", + "leven": "^3.1.0", + "pretty-format": "^29.7.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-validate/node_modules/camelcase": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz", + "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/jest-validate/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/jest-validate/node_modules/jest-get-type": { + "version": "29.6.3", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", + "integrity": "sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw==", + "dev": true, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-validate/node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-validate/node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/jest-validate/node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true + }, + "node_modules/jest-watcher": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-29.7.0.tgz", + "integrity": "sha512-49Fg7WXkU3Vl2h6LbLtMQ/HyB6rXSIX7SqvBLQmssRBGN9I0PNvPmAmCWSOY6SOvrjhI/F7/bGAv9RtnsPA03g==", + "dev": true, + "dependencies": { + "@jest/test-result": "^29.7.0", + "@jest/types": "^29.6.3", + "@types/node": "*", + "ansi-escapes": "^4.2.1", + "chalk": "^4.0.0", + "emittery": "^0.13.1", + "jest-util": "^29.7.0", + "string-length": "^4.0.1" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-watcher/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/jest-worker": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-29.7.0.tgz", + "integrity": "sha512-eIz2msL/EzL9UFTFFx7jBTkeZfku0yUAyZZZmJ93H2TYEiroIx2PQjEXcwYtYl8zXCxb+PAmA2hLIt/6ZEkPHw==", + "dev": true, + "dependencies": { + "@types/node": "*", + "jest-util": "^29.7.0", + "merge-stream": "^2.0.0", + "supports-color": "^8.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/jest-worker/node_modules/supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/supports-color?sponsor=1" + } + }, + "node_modules/jison": { + "version": "0.4.18", + "resolved": "https://registry.npmjs.org/jison/-/jison-0.4.18.tgz", + "integrity": "sha512-FKkCiJvozgC7VTHhMJ00a0/IApSxhlGsFIshLW6trWJ8ONX2TQJBBz6DlcO1Gffy4w9LT+uL+PA+CVnUSJMF7w==", + "dependencies": { + "cjson": "0.3.0", + "ebnf-parser": "0.1.10", + "escodegen": "1.3.x", + "esprima": "1.1.x", + "jison-lex": "0.3.x", + "JSONSelect": "0.4.0", + "lex-parser": "~0.1.3", + "nomnom": "1.5.2" + }, + "bin": { + "jison": "lib/cli.js" + }, + "engines": { + "node": ">=0.4" + } + }, + "node_modules/jison-lex": { + "version": "0.3.4", + "resolved": "https://registry.npmjs.org/jison-lex/-/jison-lex-0.3.4.tgz", + "integrity": "sha512-EBh5wrXhls1cUwROd5DcDHR1sG7CdsCFSqY1027+YA1RGxz+BX2TDLAhdsQf40YEtFDGoiO0Qm8PpnBl2EzDJw==", + "dependencies": { + "lex-parser": "0.1.x", + "nomnom": "1.5.2" + }, + "bin": { + "jison-lex": "cli.js" + }, + "engines": { + "node": ">=0.4" + } + }, + "node_modules/js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", + "dev": true + }, + "node_modules/js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dev": true, + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/js-yaml/node_modules/esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true, + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/jsesc": { + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", + "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", + "dev": true, + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/json-parse-even-better-errors": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", + "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", + "dev": true + }, + "node_modules/json5": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", + "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", + "dev": true, + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/jsonlint": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/jsonlint/-/jsonlint-1.6.0.tgz", + "integrity": "sha512-x6YLBe6NjdpmIeiklwQOxsZuYj/SOWkT33GlTpaG1UdFGjdWjPcxJ1CWZAX3wA7tarz8E2YHF6KiW5HTapPlXw==", + "dependencies": { + "JSV": ">= 4.0.x", + "nomnom": ">= 1.5.x" + }, + "bin": { + "jsonlint": "lib/cli.js" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/JSONSelect": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/JSONSelect/-/JSONSelect-0.4.0.tgz", + "integrity": "sha512-VRLR3Su35MH+XV2lrvh9O7qWoug/TUyj9tLDjn9rtpUCNnILLrHjgd/tB0KrhugCxUpj3UqoLqfYb3fLJdIQQQ==", + "engines": { + "node": ">=0.4.7" + } + }, + "node_modules/JSV": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/JSV/-/JSV-4.0.2.tgz", + "integrity": "sha512-ZJ6wx9xaKJ3yFUhq5/sk82PJMuUyLk277I8mQeyDgCTjGdjWJIvPfaU5LIXaMuaN2UO1X3kZH4+lgphublZUHw==", + "engines": { + "node": "*" + } + }, + "node_modules/katex": { + "version": "0.16.10", + "resolved": "https://registry.npmjs.org/katex/-/katex-0.16.10.tgz", + "integrity": "sha512-ZiqaC04tp2O5utMsl2TEZTXxa6WSC4yo0fv5ML++D3QZv/vx2Mct0mTlRx3O+uUkjfuAgOkzsCmq5MiUEsDDdA==", + "funding": [ + "https://opencollective.com/katex", + "https://github.com/sponsors/katex" + ], + "dependencies": { + "commander": "^8.3.0" + }, + "bin": { + "katex": "cli.js" + } + }, + "node_modules/katex/node_modules/commander": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-8.3.0.tgz", + "integrity": "sha512-OkTL9umf+He2DZkUq8f8J9of7yL6RJKI24dVITBmNfZBmri9zYZQrKkuXiKhyfPSu8tUhnVBB1iKXevvnlR4Ww==", + "engines": { + "node": ">= 12" + } + }, + "node_modules/khroma": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/khroma/-/khroma-2.1.0.tgz", + "integrity": "sha512-Ls993zuzfayK269Svk9hzpeGUKob/sIgZzyHYdjQoAdQetRKpOLj+k/QQQ/6Qi0Yz65mlROrfd+Ev+1+7dz9Kw==" + }, + "node_modules/kleur": { + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/kleur/-/kleur-4.1.5.tgz", + "integrity": "sha512-o+NO+8WrRiQEE4/7nwRJhN1HWpVmJm511pBHUxPLtp0BUISzlBplORYSmTclCnJvQq2tKu/sgl3xVpkc7ZWuQQ==", + "engines": { + "node": ">=6" + } + }, + "node_modules/layout-base": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/layout-base/-/layout-base-1.0.2.tgz", + "integrity": "sha512-8h2oVEZNktL4BH2JCOI90iD1yXwL6iNW7KcCKT2QZgQJR2vbqDsldCTPRU9NifTCqHZci57XvQQ15YTu+sTYPg==" + }, + "node_modules/leven": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/leven/-/leven-3.1.0.tgz", + "integrity": "sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/lex-parser": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/lex-parser/-/lex-parser-0.1.4.tgz", + "integrity": "sha512-DuAEISsr1H4LOpmFLkyMc8YStiRWZCO8hMsoXAXSbgyfvs2WQhSt0+/FBv3ZU/JBFZMGcE+FWzEBSzwUU7U27w==" + }, + "node_modules/lines-and-columns": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", + "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", + "dev": true + }, + "node_modules/locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dev": true, + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==" + }, + "node_modules/lodash-es": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash-es/-/lodash-es-4.17.21.tgz", + "integrity": "sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw==" + }, + "node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/make-dir": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-4.0.0.tgz", + "integrity": "sha512-hXdUTZYIVOt1Ex//jAQi+wTZZpUpwBj/0QsOzqegb3rGMMeJiSEu5xLHnYfBrRV4RH2+OCSOO95Is/7x1WJ4bw==", + "dev": true, + "dependencies": { + "semver": "^7.5.3" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/make-error": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", + "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", + "dev": true + }, + "node_modules/makeerror": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/makeerror/-/makeerror-1.0.12.tgz", + "integrity": "sha512-JmqCvUhmt43madlpFzG4BQzG2Z3m6tvQDNKdClZnO3VbIudJYmxsT0FNJMeiB2+JTSlTQTSbU8QdesVmwJcmLg==", + "dev": true, + "dependencies": { + "tmpl": "1.0.5" + } + }, + "node_modules/mdast-util-from-markdown": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/mdast-util-from-markdown/-/mdast-util-from-markdown-1.3.1.tgz", + "integrity": "sha512-4xTO/M8c82qBcnQc1tgpNtubGUW/Y1tBQ1B0i5CtSoelOLKFYlElIr3bvgREYYO5iRqbMY1YuqZng0GVOI8Qww==", + "dependencies": { + "@types/mdast": "^3.0.0", + "@types/unist": "^2.0.0", + "decode-named-character-reference": "^1.0.0", + "mdast-util-to-string": "^3.1.0", + "micromark": "^3.0.0", + "micromark-util-decode-numeric-character-reference": "^1.0.0", + "micromark-util-decode-string": "^1.0.0", + "micromark-util-normalize-identifier": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0", + "unist-util-stringify-position": "^3.0.0", + "uvu": "^0.5.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/mdast-util-to-string": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/mdast-util-to-string/-/mdast-util-to-string-3.2.0.tgz", + "integrity": "sha512-V4Zn/ncyN1QNSqSBxTrMOLpjr+IKdHl2v3KVLoWmDPscP4r9GcCi71gjgvUV1SFSKh92AjAG4peFuBl2/YgCJg==", + "dependencies": { + "@types/mdast": "^3.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/merge-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", + "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", + "dev": true + }, + "node_modules/mermaid": { + "version": "10.9.1", + "resolved": "https://registry.npmjs.org/mermaid/-/mermaid-10.9.1.tgz", + "integrity": "sha512-Mx45Obds5W1UkW1nv/7dHRsbfMM1aOKA2+Pxs/IGHNonygDHwmng8xTHyS9z4KWVi0rbko8gjiBmuwwXQ7tiNA==", + "dependencies": { + "@braintree/sanitize-url": "^6.0.1", + "@types/d3-scale": "^4.0.3", + "@types/d3-scale-chromatic": "^3.0.0", + "cytoscape": "^3.28.1", + "cytoscape-cose-bilkent": "^4.1.0", + "d3": "^7.4.0", + "d3-sankey": "^0.12.3", + "dagre-d3-es": "7.0.10", + "dayjs": "^1.11.7", + "dompurify": "^3.0.5", + "elkjs": "^0.9.0", + "katex": "^0.16.9", + "khroma": "^2.0.0", + "lodash-es": "^4.17.21", + "mdast-util-from-markdown": "^1.3.0", + "non-layered-tidy-tree-layout": "^2.0.2", + "stylis": "^4.1.3", + "ts-dedent": "^2.2.0", + "uuid": "^9.0.0", + "web-worker": "^1.2.0" + } + }, + "node_modules/micromark": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/micromark/-/micromark-3.2.0.tgz", + "integrity": "sha512-uD66tJj54JLYq0De10AhWycZWGQNUvDI55xPgk2sQM5kn1JYlhbCMTtEeT27+vAhW2FBQxLlOmS3pmA7/2z4aA==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "@types/debug": "^4.0.0", + "debug": "^4.0.0", + "decode-named-character-reference": "^1.0.0", + "micromark-core-commonmark": "^1.0.1", + "micromark-factory-space": "^1.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-chunked": "^1.0.0", + "micromark-util-combine-extensions": "^1.0.0", + "micromark-util-decode-numeric-character-reference": "^1.0.0", + "micromark-util-encode": "^1.0.0", + "micromark-util-normalize-identifier": "^1.0.0", + "micromark-util-resolve-all": "^1.0.0", + "micromark-util-sanitize-uri": "^1.0.0", + "micromark-util-subtokenize": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.1", + "uvu": "^0.5.0" + } + }, + "node_modules/micromark-core-commonmark": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-core-commonmark/-/micromark-core-commonmark-1.1.0.tgz", + "integrity": "sha512-BgHO1aRbolh2hcrzL2d1La37V0Aoz73ymF8rAcKnohLy93titmv62E0gP8Hrx9PKcKrqCZ1BbLGbP3bEhoXYlw==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "decode-named-character-reference": "^1.0.0", + "micromark-factory-destination": "^1.0.0", + "micromark-factory-label": "^1.0.0", + "micromark-factory-space": "^1.0.0", + "micromark-factory-title": "^1.0.0", + "micromark-factory-whitespace": "^1.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-chunked": "^1.0.0", + "micromark-util-classify-character": "^1.0.0", + "micromark-util-html-tag-name": "^1.0.0", + "micromark-util-normalize-identifier": "^1.0.0", + "micromark-util-resolve-all": "^1.0.0", + "micromark-util-subtokenize": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.1", + "uvu": "^0.5.0" + } + }, + "node_modules/micromark-factory-destination": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-factory-destination/-/micromark-factory-destination-1.1.0.tgz", + "integrity": "sha512-XaNDROBgx9SgSChd69pjiGKbV+nfHGDPVYFs5dOoDd7ZnMAE+Cuu91BCpsY8RT2NP9vo/B8pds2VQNCLiu0zhg==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-character": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-factory-label": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-factory-label/-/micromark-factory-label-1.1.0.tgz", + "integrity": "sha512-OLtyez4vZo/1NjxGhcpDSbHQ+m0IIGnT8BoPamh+7jVlzLJBH98zzuCoUeMxvM6WsNeh8wx8cKvqLiPHEACn0w==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-character": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0", + "uvu": "^0.5.0" + } + }, + "node_modules/micromark-factory-space": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-factory-space/-/micromark-factory-space-1.1.0.tgz", + "integrity": "sha512-cRzEj7c0OL4Mw2v6nwzttyOZe8XY/Z8G0rzmWQZTBi/jjwyw/U4uqKtUORXQrR5bAZZnbTI/feRV/R7hc4jQYQ==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-character": "^1.0.0", + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-factory-title": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-factory-title/-/micromark-factory-title-1.1.0.tgz", + "integrity": "sha512-J7n9R3vMmgjDOCY8NPw55jiyaQnH5kBdV2/UXCtZIpnHH3P6nHUKaH7XXEYuWwx/xUJcawa8plLBEjMPU24HzQ==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-factory-space": "^1.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-factory-whitespace": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-factory-whitespace/-/micromark-factory-whitespace-1.1.0.tgz", + "integrity": "sha512-v2WlmiymVSp5oMg+1Q0N1Lxmt6pMhIHD457whWM7/GUlEks1hI9xj5w3zbc4uuMKXGisksZk8DzP2UyGbGqNsQ==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-factory-space": "^1.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-util-character": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/micromark-util-character/-/micromark-util-character-1.2.0.tgz", + "integrity": "sha512-lXraTwcX3yH/vMDaFWCQJP1uIszLVebzUa3ZHdrgxr7KEU/9mL4mVgCpGbyhvNLNlauROiNUq7WN5u7ndbY6xg==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-util-chunked": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-chunked/-/micromark-util-chunked-1.1.0.tgz", + "integrity": "sha512-Ye01HXpkZPNcV6FiyoW2fGZDUw4Yc7vT0E9Sad83+bEDiCJ1uXu0S3mr8WLpsz3HaG3x2q0HM6CTuPdcZcluFQ==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-symbol": "^1.0.0" + } + }, + "node_modules/micromark-util-classify-character": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-classify-character/-/micromark-util-classify-character-1.1.0.tgz", + "integrity": "sha512-SL0wLxtKSnklKSUplok1WQFoGhUdWYKggKUiqhX+Swala+BtptGCu5iPRc+xvzJ4PXE/hwM3FNXsfEVgoZsWbw==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-character": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-util-combine-extensions": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-combine-extensions/-/micromark-util-combine-extensions-1.1.0.tgz", + "integrity": "sha512-Q20sp4mfNf9yEqDL50WwuWZHUrCO4fEyeDCnMGmG5Pr0Cz15Uo7KBs6jq+dq0EgX4DPwwrh9m0X+zPV1ypFvUA==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-chunked": "^1.0.0", + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-util-decode-numeric-character-reference": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-decode-numeric-character-reference/-/micromark-util-decode-numeric-character-reference-1.1.0.tgz", + "integrity": "sha512-m9V0ExGv0jB1OT21mrWcuf4QhP46pH1KkfWy9ZEezqHKAxkj4mPCy3nIH1rkbdMlChLHX531eOrymlwyZIf2iw==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-symbol": "^1.0.0" + } + }, + "node_modules/micromark-util-decode-string": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-decode-string/-/micromark-util-decode-string-1.1.0.tgz", + "integrity": "sha512-YphLGCK8gM1tG1bd54azwyrQRjCFcmgj2S2GoJDNnh4vYtnL38JS8M4gpxzOPNyHdNEpheyWXCTnnTDY3N+NVQ==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "decode-named-character-reference": "^1.0.0", + "micromark-util-character": "^1.0.0", + "micromark-util-decode-numeric-character-reference": "^1.0.0", + "micromark-util-symbol": "^1.0.0" + } + }, + "node_modules/micromark-util-encode": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-encode/-/micromark-util-encode-1.1.0.tgz", + "integrity": "sha512-EuEzTWSTAj9PA5GOAs992GzNh2dGQO52UvAbtSOMvXTxv3Criqb6IOzJUBCmEqrrXSblJIJBbFFv6zPxpreiJw==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ] + }, + "node_modules/micromark-util-html-tag-name": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/micromark-util-html-tag-name/-/micromark-util-html-tag-name-1.2.0.tgz", + "integrity": "sha512-VTQzcuQgFUD7yYztuQFKXT49KghjtETQ+Wv/zUjGSGBioZnkA4P1XXZPT1FHeJA6RwRXSF47yvJ1tsJdoxwO+Q==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ] + }, + "node_modules/micromark-util-normalize-identifier": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-normalize-identifier/-/micromark-util-normalize-identifier-1.1.0.tgz", + "integrity": "sha512-N+w5vhqrBihhjdpM8+5Xsxy71QWqGn7HYNUvch71iV2PM7+E3uWGox1Qp90loa1ephtCxG2ftRV/Conitc6P2Q==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-symbol": "^1.0.0" + } + }, + "node_modules/micromark-util-resolve-all": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-resolve-all/-/micromark-util-resolve-all-1.1.0.tgz", + "integrity": "sha512-b/G6BTMSg+bX+xVCshPTPyAu2tmA0E4X98NSR7eIbeC6ycCqCeE7wjfDIgzEbkzdEVJXRtOG4FbEm/uGbCRouA==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-types": "^1.0.0" + } + }, + "node_modules/micromark-util-sanitize-uri": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/micromark-util-sanitize-uri/-/micromark-util-sanitize-uri-1.2.0.tgz", + "integrity": "sha512-QO4GXv0XZfWey4pYFndLUKEAktKkG5kZTdUNaTAkzbuJxn2tNBOr+QtxR2XpWaMhbImT2dPzyLrPXLlPhph34A==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-character": "^1.0.0", + "micromark-util-encode": "^1.0.0", + "micromark-util-symbol": "^1.0.0" + } + }, + "node_modules/micromark-util-subtokenize": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-subtokenize/-/micromark-util-subtokenize-1.1.0.tgz", + "integrity": "sha512-kUQHyzRoxvZO2PuLzMt2P/dwVsTiivCK8icYTeR+3WgbuPqfHgPPy7nFKbeqRivBvn/3N3GBiNC+JRTMSxEC7A==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ], + "dependencies": { + "micromark-util-chunked": "^1.0.0", + "micromark-util-symbol": "^1.0.0", + "micromark-util-types": "^1.0.0", + "uvu": "^0.5.0" + } + }, + "node_modules/micromark-util-symbol": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-symbol/-/micromark-util-symbol-1.1.0.tgz", + "integrity": "sha512-uEjpEYY6KMs1g7QfJ2eX1SQEV+ZT4rUD3UcF6l57acZvLNK7PBZL+ty82Z1qhK1/yXIY4bdx04FKMgR0g4IAag==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ] + }, + "node_modules/micromark-util-types": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/micromark-util-types/-/micromark-util-types-1.1.0.tgz", + "integrity": "sha512-ukRBgie8TIAcacscVHSiddHjO4k/q3pnedmzMQ4iwDcK0FtFCohKOlFbaOL/mPgfnPsL3C1ZyxJa4sbWrBl3jg==", + "funding": [ + { + "type": "GitHub Sponsors", + "url": "https://github.com/sponsors/unifiedjs" + }, + { + "type": "OpenCollective", + "url": "https://opencollective.com/unified" + } + ] + }, + "node_modules/micromatch": { + "version": "4.0.7", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.7.tgz", + "integrity": "sha512-LPP/3KorzCwBxfeUuZmaR6bG2kdeHSbe0P2tY3FLRU4vYrjYz5hI4QZwV0njUx3jeuKe67YukQ1LSPZBKDqO/Q==", + "dev": true, + "dependencies": { + "braces": "^3.0.3", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "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/moment-mini": { + "version": "2.29.4", + "resolved": "https://registry.npmjs.org/moment-mini/-/moment-mini-2.29.4.tgz", + "integrity": "sha512-uhXpYwHFeiTbY9KSgPPRoo1nt8OxNVdMVoTBYHfSEKeRkIkwGpO+gERmhuhBtzfaeOyTkykSrm2+noJBgqt3Hg==" + }, + "node_modules/mri": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mri/-/mri-1.2.0.tgz", + "integrity": "sha512-tzzskb3bG8LvYGFF/mDTpq3jpI6Q9wc3LEmBaghu+DdCssd1FakN7Bc0hVNmEyGq1bq3RgfkCb3cmQLpNPOroA==", + "engines": { + "node": ">=4" + } + }, + "node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true + }, + "node_modules/node-int64": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", + "integrity": "sha512-O5lz91xSOeoXP6DulyHfllpq+Eg00MWitZIbtPfoSEvqIHdl5gfcY6hYzDWnj0qD5tz52PI08u9qUvSVeUBeHw==", + "dev": true + }, + "node_modules/node-releases": { + "version": "2.0.14", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.14.tgz", + "integrity": "sha512-y10wOWt8yZpqXmOgRo77WaHEmhYQYGNA6y421PKsKYWEK8aW+cqAphborZDhqfyKrbZEN92CN1X2KbafY2s7Yw==", + "dev": true + }, + "node_modules/nomnom": { + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/nomnom/-/nomnom-1.5.2.tgz", + "integrity": "sha512-fiVbT7BqxiQqjlR9U3FDGOSERFCKoXVCdxV2FwZuNN7/cmJ42iQx35nUFOAFDcyvemu9Adp+IlsCGlKQYLmBKw==", + "deprecated": "Package no longer supported. Contact support@npmjs.com for more info.", + "dependencies": { + "colors": "0.5.x", + "underscore": "1.1.x" + }, + "engines": { + "node": "*" + } + }, + "node_modules/non-layered-tidy-tree-layout": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/non-layered-tidy-tree-layout/-/non-layered-tidy-tree-layout-2.0.2.tgz", + "integrity": "sha512-gkXMxRzUH+PB0ax9dUN0yYF0S25BqeAYqhgMaLUFmpXLEk7Fcu8f4emJuOAY0V8kjDICxROIKsTAKsV/v355xw==" + }, + "node_modules/normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/npm-run-path": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-4.0.1.tgz", + "integrity": "sha512-S48WzZW777zhNIrn7gxOlISNAqi9ZC/uQFnRdbeIHhZhCA6UqpkOT8T1G7BvfdgP4Er8gF4sUbaS0i7QvIfCWw==", + "dev": true, + "dependencies": { + "path-key": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/npm-run-path/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/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dev": true, + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dev": true, + "dependencies": { + "mimic-fn": "^2.1.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "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": "4.1.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dev": true, + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/p-locate/node_modules/p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dev": true, + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-try": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", + "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/parse-json": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", + "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", + "dev": true, + "dependencies": { + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/path": { + "version": "0.12.7", + "resolved": "https://registry.npmjs.org/path/-/path-0.12.7.tgz", + "integrity": "sha512-aXXC6s+1w7otVF9UletFkFcDsJeO7lSZBPUQhtb5O0xJe8LtYhj/GxldoL09bBj9+ZmE2hNoHqQSFMN5fikh4Q==", + "dependencies": { + "process": "^0.11.1", + "util": "^0.10.3" + } + }, + "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-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "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 + }, + "node_modules/picocolors": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.1.tgz", + "integrity": "sha512-anP1Z8qwhkbmu7MFP5iTt+wQKXgwzf7zTyGlcdzabySa9vd0Xt392U0rVmz9poOaBj0uHJKyyo9/upk0HrEQew==", + "dev": true + }, + "node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/pirates": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.6.tgz", + "integrity": "sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==", + "dev": true, + "engines": { + "node": ">= 6" + } + }, + "node_modules/pkg-dir": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", + "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", + "dev": true, + "dependencies": { + "find-up": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/process": { + "version": "0.11.10", + "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz", + "integrity": "sha512-cdGef/drWFoydD1JsMzuFf8100nZl+GT+yacc2bEced5f9Rjk4z+WtFUTBu9PhOi9j/jfmBPu0mMEY4wIdAF8A==", + "engines": { + "node": ">= 0.6.0" + } + }, + "node_modules/prompts": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz", + "integrity": "sha512-NxNv/kLguCA7p3jE8oL2aEBsrJWgAakBpgmgK6lpPWV+WuOmY6r2/zbAVnP+T8bQlA0nzHXSJSJW0Hq7ylaD2Q==", + "dev": true, + "dependencies": { + "kleur": "^3.0.3", + "sisteransi": "^1.0.5" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/prompts/node_modules/kleur": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", + "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/pure-rand": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/pure-rand/-/pure-rand-6.1.0.tgz", + "integrity": "sha512-bVWawvoZoBYpp6yIoQtQXHZjmz35RSVHnUOTefl8Vcjr8snTPY1wnpSPMWekcFwbxI6gtmT7rSYPFvz71ldiOA==", + "dev": true, + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/dubzzz" + }, + { + "type": "opencollective", + "url": "https://opencollective.com/fast-check" + } + ] + }, + "node_modules/require-directory": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", + "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve": { + "version": "1.22.8", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.8.tgz", + "integrity": "sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==", + "dev": true, + "dependencies": { + "is-core-module": "^2.13.0", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/resolve-cwd": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", + "integrity": "sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg==", + "dev": true, + "dependencies": { + "resolve-from": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/resolve-cwd/node_modules/resolve-from": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", + "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/resolve.exports": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/resolve.exports/-/resolve.exports-2.0.2.tgz", + "integrity": "sha512-X2UW6Nw3n/aMgDVy+0rSqgHlv39WZAlZrXCdnbyEiKm17DSqHX4MmQMaST3FbeWR5FTuRcUwYAziZajji0Y7mg==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/robust-predicates": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/robust-predicates/-/robust-predicates-3.0.2.tgz", + "integrity": "sha512-IXgzBWvWQwE6PrDI05OvmXUIruQTcoMDzRsOd5CDvHCVLcLHMTSYvOK5Cm46kWqlV3yAbuSpBZdJ5oP5OUoStg==" + }, + "node_modules/rw": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/rw/-/rw-1.3.3.tgz", + "integrity": "sha512-PdhdWy89SiZogBLaw42zdeqtRJ//zFd2PgQavcICDUgJT5oW10QCRKbJ6bg4r0/UY2M6BWd5tkxuGFRvCkgfHQ==" + }, + "node_modules/sade": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/sade/-/sade-1.8.1.tgz", + "integrity": "sha512-xal3CZX1Xlo/k4ApwCFrHVACi9fBqJ7V+mwhBsuf/1IOKbBy098Fex+Wa/5QMubw09pSZ/u8EY8PWgevJsXp1A==", + "dependencies": { + "mri": "^1.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "node_modules/semver": { + "version": "7.6.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.2.tgz", + "integrity": "sha512-FNAIBWCx9qcRhoHcgcJ0gvU7SN1lYU2ZXuSfl04bSC5OpvDHFyJCjdNHomPXxjQlCBU67YW64PzY7/VIEH7F2w==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/signal-exit": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", + "dev": true + }, + "node_modules/sisteransi": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", + "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==", + "dev": true + }, + "node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/source-map": { + "version": "0.1.43", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.1.43.tgz", + "integrity": "sha512-VtCvB9SIQhk3aF6h+N85EaqIaBFIAfZ9Cu+NJHHVvc8BbEcnvDcFw6sqQ2dQrT6SlOrZq3tIvyD9+EGq/lJryQ==", + "optional": true, + "dependencies": { + "amdefine": ">=0.0.4" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/source-map-support": { + "version": "0.5.13", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.13.tgz", + "integrity": "sha512-SHSKFHadjVA5oR4PPqhtAVdcBWwRYVd6g6cAXnIbRiIwc2EhPrTuKUBdSLvlEKyIP3GCf89fltvcZiP9MMFA1w==", + "dev": true, + "dependencies": { + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" + } + }, + "node_modules/source-map-support/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/sprintf-js": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", + "integrity": "sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==", + "dev": true + }, + "node_modules/stack-utils": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/stack-utils/-/stack-utils-2.0.6.tgz", + "integrity": "sha512-XlkWvfIm6RmsWtNJx+uqtKLS8eqFbxUg0ZzLXqY0caEy9l7hruX8IpiDnjsLavoBgqCCR71TqWO8MaXYheJ3RQ==", + "dev": true, + "dependencies": { + "escape-string-regexp": "^2.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/stack-utils/node_modules/escape-string-regexp": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz", + "integrity": "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/string-length": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/string-length/-/string-length-4.0.2.tgz", + "integrity": "sha512-+l6rNN5fYHNhZZy41RXsYptCjA2Igmq4EG7kZAYFQI1E1VTXarr6ZPXBg6eq7Y6eK4FEhY6AJlyuFIb/v/S0VQ==", + "dev": true, + "dependencies": { + "char-regex": "^1.0.2", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/string-length/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/string-width": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dev": true, + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/string-width/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-bom": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-4.0.0.tgz", + "integrity": "sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/strip-final-newline": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-2.0.0.tgz", + "integrity": "sha512-BrpvfNAE3dcvq7ll3xVumzjKjZQ5tI1sEUIKr3Uoks0XUl45St3FlatVqef9prk4jRDzhW6WZg+3bk93y6pLjA==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/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, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/stylis": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/stylis/-/stylis-4.3.2.tgz", + "integrity": "sha512-bhtUjWd/z6ltJiQwg0dUfxEJ+W+jdqQd8TbWLWyeIJHlnsqmGLRFFd8e5mA0AZi/zx90smXRlN66YMTcaSFifg==" + }, + "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/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, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/test-exclude": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-6.0.0.tgz", + "integrity": "sha512-cAGWPIyOHU6zlmg88jwm7VRyXnMN7iV68OGAbYDk/Mh/xC/pzVPlQtY6ngoIH/5/tciuhGfvESU8GrHrcxD56w==", + "dev": true, + "dependencies": { + "@istanbuljs/schema": "^0.1.2", + "glob": "^7.1.4", + "minimatch": "^3.0.4" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/tmpl": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/tmpl/-/tmpl-1.0.5.tgz", + "integrity": "sha512-3f0uOEAQwIqGuWW2MVzYg8fV/QNnc/IpuJNG837rLuczAaLVHslWHZQj4IGiEl5Hs3kkbhwL9Ab7Hrsmuj+Smw==", + "dev": true + }, + "node_modules/to-fast-properties": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", + "integrity": "sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "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, + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/ts-dedent": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/ts-dedent/-/ts-dedent-2.2.0.tgz", + "integrity": "sha512-q5W7tVM71e2xjHZTlgfTDoPF/SmqKG5hddq9SzR49CH2hayqRKJtQ4mtRlSxKaJlR/+9rEM+mnBHf7I2/BQcpQ==", + "engines": { + "node": ">=6.10" + } + }, + "node_modules/ts-jest": { + "version": "29.1.5", + "resolved": "https://registry.npmjs.org/ts-jest/-/ts-jest-29.1.5.tgz", + "integrity": "sha512-UuClSYxM7byvvYfyWdFI+/2UxMmwNyJb0NPkZPQE2hew3RurV7l7zURgOHAd/1I1ZdPpe3GUsXNXAcN8TFKSIg==", + "dev": true, + "dependencies": { + "bs-logger": "0.x", + "fast-json-stable-stringify": "2.x", + "jest-util": "^29.0.0", + "json5": "^2.2.3", + "lodash.memoize": "4.x", + "make-error": "1.x", + "semver": "^7.5.3", + "yargs-parser": "^21.0.1" + }, + "bin": { + "ts-jest": "cli.js" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || ^18.0.0 || >=20.0.0" + }, + "peerDependencies": { + "@babel/core": ">=7.0.0-beta.0 <8", + "@jest/transform": "^29.0.0", + "@jest/types": "^29.0.0", + "babel-jest": "^29.0.0", + "jest": "^29.0.0", + "typescript": ">=4.3 <6" + }, + "peerDependenciesMeta": { + "@babel/core": { + "optional": true + }, + "@jest/transform": { + "optional": true + }, + "@jest/types": { + "optional": true + }, + "babel-jest": { + "optional": true + }, + "esbuild": { + "optional": true + } + } + }, + "node_modules/ts-jest/node_modules/lodash.memoize": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", + "integrity": "sha512-t7j+NzmgnQzTAYXcsHYLgimltOV1MXHtlOWf6GjL9Kj8GK5FInw5JotxvbOs+IvV1/Dzo04/fCGfLVs7aXb4Ag==", + "dev": true + }, + "node_modules/ts-node": { + "version": "10.9.2", + "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.9.2.tgz", + "integrity": "sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==", + "dev": true, + "dependencies": { + "@cspotcode/source-map-support": "^0.8.0", + "@tsconfig/node10": "^1.0.7", + "@tsconfig/node12": "^1.0.7", + "@tsconfig/node14": "^1.0.0", + "@tsconfig/node16": "^1.0.2", + "acorn": "^8.4.1", + "acorn-walk": "^8.1.1", + "arg": "^4.1.0", + "create-require": "^1.1.0", + "diff": "^4.0.1", + "make-error": "^1.1.1", + "v8-compile-cache-lib": "^3.0.1", + "yn": "3.1.1" + }, + "bin": { + "ts-node": "dist/bin.js", + "ts-node-cwd": "dist/bin-cwd.js", + "ts-node-esm": "dist/bin-esm.js", + "ts-node-script": "dist/bin-script.js", + "ts-node-transpile-only": "dist/bin-transpile.js", + "ts-script": "dist/bin-script-deprecated.js" + }, + "peerDependencies": { + "@swc/core": ">=1.2.50", + "@swc/wasm": ">=1.2.50", + "@types/node": "*", + "typescript": ">=2.7" + }, + "peerDependenciesMeta": { + "@swc/core": { + "optional": true + }, + "@swc/wasm": { + "optional": true + } + } + }, + "node_modules/ts-node/node_modules/acorn": { + "version": "8.12.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.12.0.tgz", + "integrity": "sha512-RTvkC4w+KNXrM39/lWCUaG0IbRkWdCv7W/IOW9oU6SawyxulvkQy5HQPVTKxEjczcUvapcrw3cFx/60VN/NRNw==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/ts-node/node_modules/acorn-walk": { + "version": "8.3.3", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.3.3.tgz", + "integrity": "sha512-MxXdReSRhGO7VlFe1bRG/oI7/mdLV9B9JJT0N8vZOhF7gFRR5l3M8W9G8JxmKV+JC5mGqJ0QvqfSOLsCPa4nUw==", + "dev": true, + "dependencies": { + "acorn": "^8.11.0" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/ts-node/node_modules/diff": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", + "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", + "dev": true, + "engines": { + "node": ">=0.3.1" + } + }, + "node_modules/type-detect": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz", + "integrity": "sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/typescript": { + "version": "5.5.2", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.5.2.tgz", + "integrity": "sha512-NcRtPEOsPFFWjobJEtfihkLCZCXZt/os3zf8nTxjVH3RvTSxjrCamJpbExGvYOF+tFHc3pA65qpdwPbzjohhew==", + "dev": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=14.17" + } + }, + "node_modules/underscore": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/underscore/-/underscore-1.1.7.tgz", + "integrity": "sha512-w4QtCHoLBXw1mjofIDoMyexaEdWGMedWNDhlWTtT1V1lCRqi65Pnoygkh6+WRdr+Bm8ldkBNkNeCsXGMlQS9HQ==", + "engines": { + "node": "*" + } + }, + "node_modules/unist-util-stringify-position": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/unist-util-stringify-position/-/unist-util-stringify-position-3.0.3.tgz", + "integrity": "sha512-k5GzIBZ/QatR8N5X2y+drfpWG8IDBzdnVj6OInRNWm1oXrzydiaAT2OQiA8DPRRZyAKb9b6I2a6PxYklZD0gKg==", + "dependencies": { + "@types/unist": "^2.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/unified" + } + }, + "node_modules/update-browserslist-db": { + "version": "1.0.16", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.0.16.tgz", + "integrity": "sha512-KVbTxlBYlckhF5wgfyZXTWnMn7MMZjMu9XG8bPlliUOP9ThaF4QnhP8qrjrH7DRzHfSk0oQv1wToW+iA5GajEQ==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ], + "dependencies": { + "escalade": "^3.1.2", + "picocolors": "^1.0.1" + }, + "bin": { + "update-browserslist-db": "cli.js" + }, + "peerDependencies": { + "browserslist": ">= 4.21.0" + } + }, + "node_modules/util": { + "version": "0.10.4", + "resolved": "https://registry.npmjs.org/util/-/util-0.10.4.tgz", + "integrity": "sha512-0Pm9hTQ3se5ll1XihRic3FDIku70C+iHUdT/W926rSgHV5QgXsYbKZN8MSC3tJtSkhuROzvsQjAaFENRXr+19A==", + "dependencies": { + "inherits": "2.0.3" + } + }, + "node_modules/uuid": { + "version": "9.0.1", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-9.0.1.tgz", + "integrity": "sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA==", + "funding": [ + "https://github.com/sponsors/broofa", + "https://github.com/sponsors/ctavan" + ], + "bin": { + "uuid": "dist/bin/uuid" + } + }, + "node_modules/uvu": { + "version": "0.5.6", + "resolved": "https://registry.npmjs.org/uvu/-/uvu-0.5.6.tgz", + "integrity": "sha512-+g8ENReyr8YsOc6fv/NVJs2vFdHBnBNdfE49rshrTzDWOlUx4Gq7KOS2GD8eqhy2j+Ejq29+SbKH8yjkAqXqoA==", + "dependencies": { + "dequal": "^2.0.0", + "diff": "^5.0.0", + "kleur": "^4.0.3", + "sade": "^1.7.3" + }, + "bin": { + "uvu": "bin.js" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/v8-compile-cache-lib": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz", + "integrity": "sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==", + "dev": true + }, + "node_modules/v8-to-istanbul": { + "version": "9.3.0", + "resolved": "https://registry.npmjs.org/v8-to-istanbul/-/v8-to-istanbul-9.3.0.tgz", + "integrity": "sha512-kiGUalWN+rgBJ/1OHZsBtU4rXZOfj/7rKQxULKlIzwzQSvMJUUNgPwJEEh7gU6xEVxC0ahoOBvN2YI8GH6FNgA==", + "dev": true, + "dependencies": { + "@jridgewell/trace-mapping": "^0.3.12", + "@types/istanbul-lib-coverage": "^2.0.1", + "convert-source-map": "^2.0.0" + }, + "engines": { + "node": ">=10.12.0" + } + }, + "node_modules/v8-to-istanbul/node_modules/convert-source-map": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz", + "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", + "dev": true + }, + "node_modules/walker": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/walker/-/walker-1.0.8.tgz", + "integrity": "sha512-ts/8E8l5b7kY0vlWLewOkDXMmPdLcVV4GmOQLyxuSswIJsweeFZtAsMF7k1Nszz+TYBQrlYRmzOnr398y1JemQ==", + "dev": true, + "dependencies": { + "makeerror": "1.0.12" + } + }, + "node_modules/web-worker": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/web-worker/-/web-worker-1.3.0.tgz", + "integrity": "sha512-BSR9wyRsy/KOValMgd5kMyr3JzpdeoR9KVId8u5GVlTTAtNChlsE4yTxeY7zMdNSyOmoKBv8NH2qeRY9Tg+IaA==" + }, + "node_modules/wrap-ansi": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/wrap-ansi/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dev": true + }, + "node_modules/write-file-atomic": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-4.0.2.tgz", + "integrity": "sha512-7KxauUdBmSdWnmpaGFg+ppNjKF8uNLry8LyzjauQDOVONfFLNKrKvQOxZ/VuTIcS/gge/YNahf5RIIQWTSarlg==", + "dev": true, + "dependencies": { + "imurmurhash": "^0.1.4", + "signal-exit": "^3.0.7" + }, + "engines": { + "node": "^12.13.0 || ^14.15.0 || >=16.0.0" + } + }, + "node_modules/xml": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/xml/-/xml-1.0.1.tgz", + "integrity": "sha512-huCv9IH9Tcf95zuYCsQraZtWnJvBtLVE0QHMOs8bWyZAFZNDcYjsPq1nEx8jKA9y+Beo9v+7OBPRisQTjinQMw==", + "dev": true + }, + "node_modules/y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true + }, + "node_modules/yargs": { + "version": "17.7.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", + "dev": true, + "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" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/yargs-parser": { + "version": "21.1.1", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz", + "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==", + "dev": true, + "engines": { + "node": ">=12" + } + }, + "node_modules/yn": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", + "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 0000000..d6c5380 --- /dev/null +++ b/package.json @@ -0,0 +1,53 @@ +{ + "name": "sqltooling-drawio", + "version": "0.0.1", + "description": "plugins for sql tooling in drawio", + "main": "index.js", + "engines": { + "node": ">=16.x" + }, + "scripts": { + "start": "node index.mjs", + "test": "jest -c ./jest.config.ts --forceExit --verbose -i --no-cache --detectOpenHandles", + "test:coverage": "jest --forceExit --coverage --verbose --detectOpenHandles", + "test:watch": "jest --watchAll --detectOpenHandles", + "lint": "tsc --noEmit && eslint \"{src,client}/**/*.{js,ts}\"", + "lint:fix": "tsc --noEmit && eslint \"{src,client}/**/*.{js,ts}\" --fix", + "create": "npm run build && npm run cli:test", + "build": "tsc && npm run build:types", + "build:types": "tsc -p tsconfig-types.json", + "build:help": "tsc --help", + "build:client": "npm run mkdir:dist && browserify src/index.ts -p [ tsify --noImplicitAny ] > dist/littleMermaid.js", + "mkdir:dist": "node ./build/createDistFolder.js" + }, + "repository": { + "type": "git", + "url": "git+https://github.com/lastlink/sqltooling-drawio.git" + }, + "keywords": [ + "uml-diagram", + "drawio", + "sqltools" + ], + "author": "lastlink", + "license": "MIT", + "bugs": { + "url": "https://github.com/lastlink/sqltooling-drawio/issues" + }, + "homepage": "https://github.com/lastlink/sqltooling-drawio#readme", + "dependencies": { + "@funktechno/little-mermaid-2-the-sql": "^0.1.1" + }, + "devDependencies": { + "jest": "^29.0.3", + "jest-junit": "^16.0.0", + "jest-serial-runner": "^1.1.0", + "ts-jest": "^29.0.1", + "ts-node": "^10.9.1", + "typescript": "^5.0.2" + }, + "directories": { + "lib": "lib", + "test": "tests" + } +} diff --git a/src/sql.ts b/src/sql.ts new file mode 100644 index 0000000..75f85a2 --- /dev/null +++ b/src/sql.ts @@ -0,0 +1,1290 @@ +import { DbParser } from "@funktechno/little-mermaid-2-the-sql/lib/src/generate-sql-ddl"; +import { DbDefinition, DbRelationshipDefinition } from "@funktechno/little-mermaid-2-the-sql/lib/src/types"; +import { ColumnQuantifiers, TableAttribute, TableEntity } from "./types/sql-plugin-types"; + +/** + * Parse SQL CREATE TABLE. Simple initial version for community to improve. + */ +Draw.loadPlugin(function(ui) { + + // export sql methods + + /** + * Mermaid Models TO SQL parser + * src https://github.com/Software-Developers-IRL/Little-Mermaid-2-The-SQL/blob/main/src/generate-sql-ddl.ts + */ + // DbParser + + //Create Base div + const divGenSQL = document.createElement('div'); + divGenSQL.style.userSelect = 'none'; + divGenSQL.style.overflow = 'hidden'; + divGenSQL.style.padding = '10px'; + divGenSQL.style.height = '100%'; + + const sqlInputGenSQL = document.createElement('textarea'); + sqlInputGenSQL.style.height = '200px'; + sqlInputGenSQL.style.width = '100%'; + sqlInputGenSQL.value = '-- click a database type button' + mxUtils.br(divGenSQL); + divGenSQL.appendChild(sqlInputGenSQL); + var theMenuExportAs = ui.menus.get('exportAs'); + let buttonLabel = 'tosql=To SQL' + // vscode extension support + if(!(theMenuExportAs && theMenuExportAs.enabled)) { + buttonLabel = 'tosql=Export As SQL' + } + // Extends Extras menu + mxResources.parse(buttonLabel); + + const wndGenSQL = new mxWindow(mxResources.get('tosql'), divGenSQL, document.body.offsetWidth - 480, 140, + 320, 320, true, true); + wndGenSQL.destroyOnClose = false; + wndGenSQL.setMaximizable(false); + wndGenSQL.setResizable(false); + wndGenSQL.setClosable(true); + + /** + * return text quantifiers for dialect + * @returns json + */ + function GetColumnQuantifiers(type:string):ColumnQuantifiers { + let chars = { + Start: '"', + End: '"', + }; + if (type == "mysql") { + chars.Start = "`"; + chars.End = "`"; + } + else if (type == "sqlserver") { + chars.Start = "["; + chars.End = "]"; + } + return chars; + } + /** + * sometimes rows have spans or styles, an attempt to remove them + * @param {*} label + * @returns + */ + function removeHtml(label:string){ + var div = document.createElement("div"); + divGenSQL.innerHTML = label; + var text = div.textContent || div.innerText || ""; + return text; + } + /** + * extract row column attributes + * @param {*} label + * @param {*} columnQuantifiers + * @returns + */ + function getDbLabel(label:string, columnQuantifiers:ColumnQuantifiers): TableAttribute{ + label = removeHtml(label) + // fix duplicate spaces and different space chars + label = label + .replace(/\s+/g, " ") + let firstSpaceIndex = label[0] == columnQuantifiers.Start && + label.indexOf(columnQuantifiers.End + " ") !== -1 + ? label.indexOf(columnQuantifiers.End + " ") + : label.indexOf(" "); + let attributeType = label.substring(firstSpaceIndex + 1).trim(); + let attributeName = RemoveNameQuantifiers(label.substring(0, firstSpaceIndex)); + let attribute = { + attributeName, + attributeType + } + return attribute + } + function RemoveNameQuantifiers(name:string) { + return name.replace(/\[|\]|\(|\"|\'|\`/g, "").trim(); + } + + function getMermaidDiagramDb(type: string):DbDefinition{ + var model = ui.editor.graph.getModel() + // same models from mermaid for diagram relationships + // only difference is entities is an array rather than object to allow duplicate tables + let entities: Record = {} + let relationships:DbRelationshipDefinition[] = [] + // build models + for (const key in model.cells) { + if (Object.hasOwnProperty.call(model.cells, key)) { + const mxcell = model.cells[key]; + if(mxcell.mxObjectId.indexOf("mxCell") !== -1) { + if(mxcell.style && mxcell.style.trim().startsWith("swimlane;")){ + let entity:TableEntity = { + name: RemoveNameQuantifiers(mxcell.value), + attributes: [] as TableAttribute[], + } + for (let c = 0; c < mxcell.children.length; c++) { + const col = mxcell.children[c]; + if(col.mxObjectId.indexOf("mxCell") !== -1) { + if(col.style && col.style.trim().startsWith("shape=partialRectangle")){ + const columnQuantifiers = GetColumnQuantifiers(type); + //Get delimiter of column name + //Get full name + let attribute = getDbLabel(col.value, columnQuantifiers) + var attributeKeyType = col.children.find(x=> ["FK","PK"].findIndex(k => k== x.value.toUpperCase()) !== -1 || + x.value.toUpperCase().indexOf("PK,")!=-1) + if(attributeKeyType){ + attribute.attributeKeyType = attributeKeyType.value + if(attribute.attributeKeyType != "PK" && attribute.attributeKeyType.indexOf("PK") != -1){ + attribute.attributeKeyType = "PK" + } + } + entity.attributes.push(attribute) + if(col.edges && col.edges.length){ + // check for edges foreign keys + for (let e = 0; e < col.edges.length; e++) { + const edge = col.edges[e]; + if(edge.mxObjectId.indexOf("mxCell") !== -1) { + if(edge.style && edge.style.indexOf("endArrow=") != -1 && edge.source && + edge.source.value && edge.target && edge.target.value){ + // need to check if end is open or certain value to determin relationship type + // extract endArrow txt + // check if both match and contain many or open + // if both match and are many then create a new table + let endCheck = "endArrow=" + let endArr = edge.style.indexOf(endCheck) != -1 ? + edge.style.substring(edge.style.indexOf(endCheck) + endCheck.length, edge.style.substring(edge.style.indexOf(endCheck) + endCheck.length).indexOf(";") + edge.style.indexOf(endCheck) + endCheck.length) + : "" + let startCheck = "startArrow=" + let startArr = edge.style.indexOf(startCheck) != -1 ? + edge.style.substring(edge.style.indexOf(startCheck) + startCheck.length, edge.style.substring(edge.style.indexOf(startCheck) + startCheck.length).indexOf(";") + edge.style.indexOf(startCheck) + startCheck.length) + : "" + + var manyCheck = ["open","many"] + var sourceIsPrimary = endArr && manyCheck + .findIndex(x => endArr.toLocaleLowerCase().indexOf(x)!=-1) != -1; + var targetIsPrimary = startArr && manyCheck + .findIndex(x => startArr.toLocaleLowerCase().indexOf(x)!=-1) != -1; + // has to be one to many and not one to one + if((targetIsPrimary || sourceIsPrimary) && + !(targetIsPrimary && sourceIsPrimary) + ){ + var sourceId = edge.source.value; + var sourceAttr = getDbLabel(sourceId, columnQuantifiers); + sourceId = sourceAttr.attributeName + var sourceEntity = RemoveNameQuantifiers(edge.source.parent.value); + var targetId = edge.target.value; + var targetAttr = getDbLabel(targetId, columnQuantifiers); + targetId = targetAttr.attributeName + var targetEntity = RemoveNameQuantifiers(edge.target.parent.value); + // entityA primary + // entityB foreign + let relationship:DbRelationshipDefinition = { + entityA: sourceIsPrimary ? sourceEntity : targetEntity, + entityB: sourceIsPrimary ? targetEntity : sourceEntity, + // based off of styles? + relSpec: { + cardA: 'ZERO_OR_MORE', + cardB: 'ONLY_ONE', + relType: "IDENTIFYING" + }, + roleA: sourceIsPrimary ? + `[${sourceEntity}.${sourceId}] to [${targetEntity}.${targetId}]` : + `[${targetEntity}.${targetId}] to [${sourceEntity}.${sourceId}]` + } + // check that is doesn't already exist + var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA) + if(exists ==-1){ + relationships.push(relationship) + } + } else if(targetIsPrimary && sourceIsPrimary){ + // add a new many to many table + var sourceId = edge.source.value; + sourceAttr = getDbLabel(sourceId, columnQuantifiers); + sourceAttr.attributeKeyType = "PK" + sourceId = sourceAttr.attributeName + var sourceEntity = RemoveNameQuantifiers(edge.source.parent.value); + var targetId = edge.target.value; + targetAttr = getDbLabel(targetId, columnQuantifiers); + targetAttr.attributeKeyType = "PK" + targetId = targetAttr.attributeName + var targetEntity = RemoveNameQuantifiers(edge.target.parent.value); + let compositeEntity = { + name: RemoveNameQuantifiers(sourceEntity) + "_" + RemoveNameQuantifiers(targetEntity), + attributes: [sourceAttr, targetAttr] + } + // add composite entity + if(entities[compositeEntity.name]){ + // DON'T add duplicate composite tables + } else { + entities[compositeEntity.name] = compositeEntity + } + // entityA primary + // entityB foreign + let relationship = { + entityA: sourceEntity, + entityB: compositeEntity.name, + // based off of styles? + relSpec: { + cardA: 'ZERO_OR_MORE', + cardB: 'ONLY_ONE', + relType: "IDENTIFYING" + }, + roleA: `[${sourceEntity}.${sourceId}] to [${compositeEntity.name}.${sourceId}]` + } + // check that is doesn't already exist + var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA) + if(exists ==-1){ + relationships.push(relationship) + } + let relationship2 = { + entityA: targetEntity, + entityB: compositeEntity.name, + // based off of styles? + relSpec: { + cardA: 'ZERO_OR_MORE', + cardB: 'ONLY_ONE', + relType: "IDENTIFYING" + }, + roleA: `[${targetEntity}.${targetId}] to [${compositeEntity.name}.${targetId}]` + } + // check that is doesn't already exist + exists = relationships.findIndex(r => r.entityA == relationship2.entityA && r.entityB == relationship2.entityB && r.roleA == relationship2.roleA) + if(exists ==-1){ + relationships.push(relationship2) + } + } + + } + } + + } + } + } + } + } + // allows for duplicates if another table has the same name + if(entities[entity.name]){ + var count = 2; + while(entities[entity.name + count.toString()]){ + count++; + } + entities[entity.name + count.toString()] = entity + } else { + entities[entity.name] = entity + } + } + + } + } + } + + class DatabaseModel{ + constructor(entities: Record, relationships:DbRelationshipDefinition[]){ + this.entities = entities; + this.relationships = relationships; + } + + private entities: Record; + + private relationships: DbRelationshipDefinition[]; + + getEntities(){ + return this.entities + } + + getRelationships(){ + return this.relationships + } + } + + var db = new DatabaseModel(entities, relationships) as unknown as DbDefinition; + + return db; + } + + function generateSql(type: string) { + + // get diagram model + var db = getMermaidDiagramDb(type); + // load parser + var parser = new DbParser(type, db) + // generate sql + var sql = parser.getSQLDataDefinition() + sql = `/*\n\tGenerated in drawio\n\tDatabase: ${type}\n*/\n\n` + sql + sql = sql.trim(); + // update sql value in text area + sqlInputGenSQL.value = sql; + // TODO: use selection as well? + var modelSelected = ui.editor.graph.getSelectionModel() + }; + + mxUtils.br(divGenSQL); + + const resetBtnGenSQL = mxUtils.button(mxResources.get('reset'), function() { + sqlInputGenSQL.value = ''; + }); + + resetBtnGenSQL.style.marginTop = '8px'; + resetBtnGenSQL.style.marginRight = '4px'; + resetBtnGenSQL.style.padding = '4px'; + divGenSQL.appendChild(resetBtnGenSQL); + + const btnGenSQL_mysql = mxUtils.button('MySQL', function() { + generateSql('mysql'); + }); + + btnGenSQL_mysql.style.marginTop = '8px'; + btnGenSQL_mysql.style.padding = '4px'; + divGenSQL.appendChild(btnGenSQL_mysql); + + const btnGenSQL_sqlserver = mxUtils.button('SQL Server', function() { + generateSql('sqlserver'); + }); + + btnGenSQL_sqlserver.style.marginTop = '8px'; + btnGenSQL_sqlserver.style.padding = '4px'; + divGenSQL.appendChild(btnGenSQL_sqlserver); + + const btnGenSQL_postgres = mxUtils.button('PostgreSQL', function() { + generateSql('postgres'); + }); + + btnGenSQL_postgres.style.marginTop = '8px'; + btnGenSQL_postgres.style.padding = '4px'; + divGenSQL.appendChild(btnGenSQL_postgres); + + const btnGenSQL_sqlite = mxUtils.button('Sqlite', function() { + generateSql('sqlite'); + }); + + btnGenSQL_sqlite.style.marginTop = '8px'; + btnGenSQL_sqlite.style.padding = '4px'; + divGenSQL.appendChild(btnGenSQL_sqlite); + + // Adds action + ui.actions.addAction('tosql', function() { + wndGenSQL.setVisible(!wndGenSQL.isVisible()); + + if (wndGenSQL.isVisible()) { + sqlInputGenSQL.focus(); + } + }); + // end export sql methods + + // import diagrams from sql text methods + + const contants_1 ={} + contants_1.CONSTRAINT_Foreign_Key = contants_1.CONSTRAINT_Primary_Key = contants_1.Foreign_Key = contants_1.Primary_Key = contants_1.CONSTRAINT = contants_1.AlterTable = contants_1.CreateTable = void 0; + contants_1.CreateTable = "create table"; + contants_1.AlterTable = "alter table "; + contants_1.CONSTRAINT = "constraint"; + contants_1.Primary_Key = "primary key"; + contants_1.Foreign_Key = "foreign key"; + contants_1.CONSTRAINT_Primary_Key = "constraint primary key"; + contants_1.CONSTRAINT_Foreign_Key = "constraint foreign key"; + + /** + * Main Parser class + */ + class SqlSimpleParser { + /** + * Parser constructor. + * Default dialect is 'sqlite'. + * Options: "mysql" | "sqlite" | "postgres" | "sqlserver" . + * + * @param dialect SQL dialect ('sqlite'). + */ + constructor(dialect = "sqlite") { + this.tableList = []; + /** + * Parsed statements. + */ + this.statements = []; + /** + * Remains of string feed, after last parsed statement. + */ + this.remains = ""; + /** + * Whether preparser is currently escaped. + */ + this.escaped = false; + /** + * Current quote char of preparser. + */ + this.quoted = ""; + this.exportedTables = 0; + this.SQLServer = "sqlserver"; + this.MODE_SQLSERVER = false; + this.foreignKeyList = []; + this.primaryKeyList = []; + this.dialect = dialect; + this.MODE_SQLSERVER = + this.dialect !== undefined && + this.dialect !== null && + this.dialect == this.SQLServer; + } + /** + * Feed chunk of string into parser. + * + * @param chunk Chunk of string to be parsed. + */ + feed(chunk) { + // + const removedComments = chunk + // remove database comments, multiline, --, and // + .replace(/\/\*[\s\S]*?\*\/|\/\/|--.*/g, "") + .replace(/IF NOT EXISTS/gi, "") + .trim(); + const cleanedLines = removedComments + .split("\n") + // remove empty lines + .filter((n) => n) + // remove multiple spaces + .map((n) => n.replace(/\s+/g, " ").trim()); + // combine lines that are in parenthesis + const lines = []; + let insertSameLine = false; + cleanedLines.forEach((n) => { + if (lines.length > 0){ + if((n[0] == "(" && + lines[lines.length - 1].toLocaleLowerCase().indexOf(contants_1.CreateTable) == + -1) || + insertSameLine) { + if (lines.length > 0) { + insertSameLine = true; + lines[lines.length - 1] += ` ${n}`; + if (n[0] == ")") + insertSameLine = false; + } + } + else if(lines[lines.length - 1].match(/CONSTRAINT/gi) && + (n.match(/FOREIGN KEY/gi) && !n.match(/CONSTRAINT/gi)) + ){ + lines[lines.length - 1] += ` ${n}`; + } + // add to previous line if current has references and previous has foreign key + else if(lines[lines.length - 1].match(/FOREIGN KEY/gi) && + (n.match(/REFERENCES/gi) && !n.match(/FOREIGN KEY/gi)) + ){ + lines[lines.length - 1] += ` ${n}`; + } + else if(n.substring(0,2).toUpperCase() == "ON"){ + lines[lines.length - 1] += ` ${n}`; + } + else { + lines.push(n); + } + } + else { + lines.push(n); + } + }); + let currentTableModel = null; + //Parse SQL to objects + for (let i = 0; i < lines.length; i++) { + // rowCell = null; + const tmp = lines[i].trim(); + const propertyRow = tmp.toLowerCase().trim(); + if (propertyRow[0] == ")") { + // close table + if (currentTableModel) { + this.tableList.push(currentTableModel); + currentTableModel = null; + } + continue; + } + //Parse Table + if (propertyRow.indexOf(contants_1.CreateTable) != -1) { + //Parse row + let name = tmp + .replace(this.stringToRegex(`/${contants_1.CreateTable}/gi`), "") + .trim(); + //Parse Table Name + name = this.ParseTableName(name); + if (currentTableModel !== null) { + //Add table to the list + this.tableList.push(currentTableModel); + } + //Create Table + currentTableModel = this.CreateTable(name); + } + // Parse Properties + else if (tmp !== "(" && + currentTableModel != null && + propertyRow.indexOf(contants_1.AlterTable) == -1) { + //Parse the row + let name = tmp.substring(0, tmp.charAt(tmp.length - 1) === "," ? tmp.length - 1 : tmp.length); + //Attempt to get the Key Type + let propertyType = name.toLowerCase(); + // .substring(0, AlterTable.length).toLowerCase(); + //Add special constraints + if (this.MODE_SQLSERVER) { + if (propertyType.indexOf(contants_1.CONSTRAINT) !== -1 && + propertyType.indexOf(contants_1.Primary_Key) !== -1) { + propertyType = contants_1.CONSTRAINT_Primary_Key; + } + if (propertyType.indexOf(contants_1.CONSTRAINT) !== -1 && + propertyType.indexOf(contants_1.Foreign_Key) !== -1) { + propertyType = contants_1.CONSTRAINT_Foreign_Key; + } + } + //Verify if this is a property that doesn't have a relationship (One minute of silence for the property) + const normalProperty = !propertyType.match(/PRIMARY KEY\s?\(/gi) && + propertyType.indexOf(contants_1.Foreign_Key) == -1 && + propertyType.indexOf(contants_1.CONSTRAINT_Primary_Key) == -1 && + propertyType.indexOf(contants_1.CONSTRAINT_Foreign_Key) == -1; + const nameSkipCheck = name.toUpperCase().trim(); + //Parse properties that don't have relationships + if (normalProperty) { + if (name === "" || name === "" || name === ");") { + continue; + } + let ExtendedProperties = null; + if (this.MODE_SQLSERVER) { + if (nameSkipCheck.indexOf(" ASC") !== -1 || + nameSkipCheck.indexOf(" DESC") !== -1 || + nameSkipCheck.indexOf("EXEC ") !== -1 || + nameSkipCheck.indexOf("WITH ") !== -1 || + nameSkipCheck.indexOf(" ON") !== -1 || + nameSkipCheck.indexOf("ALTER ") !== -1 || + // comments already removed + nameSkipCheck.indexOf("/*") !== -1 || + nameSkipCheck.indexOf(" CONSTRAINT") !== -1 || + nameSkipCheck.indexOf("SET ") !== -1 || + nameSkipCheck.indexOf(" NONCLUSTERED") !== -1 || + // no spaces desired + nameSkipCheck.indexOf("GO") !== -1 || + nameSkipCheck.indexOf("REFERENCES ") !== -1) { + continue; + } + //Get delimiter of column name + const firstSpaceIndex = name[0] == "[" && name.indexOf("]" + " ") !== -1 + ? name.indexOf("]" + " ") + : name.indexOf(" "); + ExtendedProperties = name.substring(firstSpaceIndex + 1).trim(); + //Get full name + name = name.substring(0, firstSpaceIndex); + name = this.RemoveNameQuantifiers(name); + } + else { + const columnQuantifiers = this.GetColumnQuantifiersFromDialect(); + //Get delimiter of column name + const firstSpaceIndex = name[0] == columnQuantifiers.Start && + name.indexOf(columnQuantifiers.End + " ") !== -1 + ? name.indexOf(columnQuantifiers.End + " ") + : name.indexOf(" "); + ExtendedProperties = name.substring(firstSpaceIndex + 1).trim(); + //Get full name + name = name.substring(0, firstSpaceIndex); + name = this.RemoveNameQuantifiers(name); + } + //Create Property + const propertyModel = this.CreateProperty(name, currentTableModel.Name, null, false, ExtendedProperties); + //Add Property to table + currentTableModel.Properties.push(propertyModel); + if (ExtendedProperties.toLocaleLowerCase().indexOf(contants_1.Primary_Key) > -1) { + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(name, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + } + } + else { + //Parse Primary Key + if (propertyType.indexOf(contants_1.Primary_Key) != -1 || + propertyType.indexOf(contants_1.CONSTRAINT_Primary_Key) != -1) { + if (!this.MODE_SQLSERVER) { + const primaryKey = name + .replace(/PRIMARY KEY\s?\(/gi, "") + .replace(")", ""); + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(primaryKey, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + } + else { + if (propertyRow.indexOf(contants_1.Primary_Key) !== -1 && + nameSkipCheck.indexOf("CLUSTERED") === -1) { + const primaryKey = name + .replace(/PRIMARY KEY\s?\(/gi, "") + .replace(")", ""); + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(primaryKey, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + } + else { + const startIndex = name.toLocaleLowerCase().indexOf("("); + const endIndex = name.indexOf(")") + 1; + const primaryKey = name + .substring(startIndex, endIndex) + .replace("(", "") + .replace(")", "") + .replace(/ASC/gi, "") + .trim(); + const columnQuantifiers = this.GetColumnQuantifiersFromDialect(); + //Get delimiter of column name + const firstSpaceIndex = primaryKey[0] == columnQuantifiers.Start && + primaryKey.indexOf(columnQuantifiers.End + " ") !== -1 + ? primaryKey.indexOf(columnQuantifiers.End + " ") + : primaryKey.indexOf(" "); + const primaryKeyRow = firstSpaceIndex == -1 + ? primaryKey + : primaryKey.substring(firstSpaceIndex + 1).trim(); + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(primaryKeyRow, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + } + } + } + } + //Parse Foreign Key + if (propertyType.indexOf(contants_1.Foreign_Key) != -1 || + propertyType.indexOf(contants_1.CONSTRAINT_Foreign_Key) != -1) { + if (!this.MODE_SQLSERVER || propertyRow.indexOf(contants_1.AlterTable) == -1) { + this.ParseMySQLForeignKey(name, currentTableModel); + } + else { + let completeRow = name; + if (nameSkipCheck.indexOf("REFERENCES") === -1) { + const referencesRow = lines[i + 1].trim(); + completeRow = + "ALTER TABLE [dbo].[" + + currentTableModel.Name + + "] WITH CHECK ADD" + + " " + + name + + " " + + referencesRow; + } + this.ParseSQLServerForeignKey(completeRow, currentTableModel); + } + } + } + else if (propertyRow.indexOf(contants_1.AlterTable) != -1) { + if (this.MODE_SQLSERVER) { + //Parse the row + const alterTableRow = tmp.substring(0, tmp.charAt(tmp.length - 1) === "," ? tmp.length - 1 : tmp.length); + const referencesRow = lines[i + 1].trim(); + const completeRow = alterTableRow + " " + referencesRow; + this.ParseSQLServerForeignKey(completeRow, currentTableModel); + } + } + } + // parse fk and primary keys + if (this.primaryKeyList.length > 0) { + this.primaryKeyList.forEach((pk) => { + // find table index + const pkTableIndex = this.tableList.findIndex((t) => t.Name.toLocaleLowerCase() == + pk.PrimaryKeyTableName.toLocaleLowerCase()); + // find property index + if (pkTableIndex > -1) { + const propertyIndex = this.tableList[pkTableIndex].Properties.findIndex((p) => p.Name.toLocaleLowerCase() == + pk.PrimaryKeyName.toLocaleLowerCase()); + if (propertyIndex > -1) { + this.tableList[pkTableIndex].Properties[propertyIndex].IsPrimaryKey = true; + } + } + }); + } + if (this.foreignKeyList.length > 0) { + this.foreignKeyList.forEach((fk) => { + // find table index + const pkTableIndex = this.tableList.findIndex((t) => t.Name.toLocaleLowerCase() == + fk.ReferencesTableName.toLocaleLowerCase()); + // find property index + if (pkTableIndex > -1) { + const propertyIndex = this.tableList[pkTableIndex].Properties.findIndex((p) => p.Name.toLocaleLowerCase() == + fk.PrimaryKeyName.toLocaleLowerCase()); + if (propertyIndex > -1) { + this.tableList[pkTableIndex].Properties[propertyIndex].ForeignKey.push(fk); + if (!fk.IsDestination) { + this.tableList[pkTableIndex].Properties[propertyIndex].IsForeignKey = true; + } + } + } + }); + } + return this; + } + stringToRegex(str) { + // Main regex + const mainResult = str.match(/\/(.+)\/.*/); + const optionsResult = str.match(/\/.+\/(.*)/); + if (mainResult && optionsResult) { + const main = mainResult[1]; + // Regex options + const options = optionsResult[1]; + // Compiled regex + return new RegExp(main, options); + } + return new RegExp("//(.+)/.*/", "//.+/(.*)/"); + } + CreatePrimaryKey(primaryKeyName, primaryKeyTableName) { + const primaryKeyNames = this.RemoveNameQuantifiers(primaryKeyName) + .split(",") + .filter((n) => n) + // remove multiple spaces + .map((n) => n.replace(/\s+/g, " ").trim()); + const primaryKeys = []; + primaryKeyNames.forEach(name => { + const primaryKey = { + PrimaryKeyTableName: primaryKeyTableName, + PrimaryKeyName: name, + }; + primaryKeys.push(primaryKey); + }); + return primaryKeys; + } + CreateProperty(name, tableName, foreignKey, isPrimaryKey, columnProps) { + const isForeignKey = foreignKey !== undefined && foreignKey !== null; + const property = { + Name: name, + ColumnProperties: columnProps, + TableName: tableName, + ForeignKey: foreignKey || [], + IsForeignKey: isForeignKey, + IsPrimaryKey: isPrimaryKey, + }; + return property; + } + ParseMySQLForeignKey(name, currentTableModel) { + const referencesIndex = name.toLowerCase().indexOf("references"); + let foreignKeySQL = name.substring(0, referencesIndex); + foreignKeySQL = foreignKeySQL.substring(foreignKeySQL.toUpperCase().indexOf("FOREIGN KEY")) + let referencesSQL = name.substring(referencesIndex, name.length); + //Remove references syntax + referencesSQL = referencesSQL.replace(/REFERENCES /gi, ""); + //Get Table and Property Index + const referencedTableIndex = referencesSQL.indexOf("("); + const referencedPropertyIndex = referencesSQL.indexOf(")"); + //Get Referenced Table + const referencedTableName = referencesSQL.substring(0, referencedTableIndex); + //Get Referenced Key + const referencedPropertyName = referencesSQL.substring(referencedTableIndex + 1, referencedPropertyIndex); + //Get ForeignKey + const foreignKey = foreignKeySQL + .replace(/FOREIGN KEY\s?\(/gi, "") + .replace(")", "") + .replace(" ", ""); + //Create ForeignKey + const foreignKeyOriginModel = this.CreateForeignKey(foreignKey, currentTableModel.Name, referencedPropertyName, referencedTableName, true); + //Add ForeignKey Origin + this.foreignKeyList.push(foreignKeyOriginModel); + //Create ForeignKey + const foreignKeyDestinationModel = this.CreateForeignKey(referencedPropertyName, referencedTableName, foreignKey, currentTableModel.Name, false); + //Add ForeignKey Destination + this.foreignKeyList.push(foreignKeyDestinationModel); + } + ParseSQLServerForeignKey(name, currentTableModel) { + const referencesIndex = name.toLowerCase().indexOf("references"); + let foreignKeySQL = ""; + if (name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`) !== -1) { + foreignKeySQL = name + .substring(name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`), referencesIndex) + .replace(/FOREIGN KEY\(/gi, "") + .replace(")", ""); + } + else { + foreignKeySQL = name + .substring(name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`), referencesIndex) + .replace(/FOREIGN KEY\s?\(/gi, "") + .replace(")", ""); + } + let referencesSQL = name.substring(referencesIndex, name.length); + const nameSkipCheck = name.toUpperCase().trim(); + let alterTableName = name + .substring(0, nameSkipCheck.indexOf("WITH")) + .replace(/ALTER TABLE /gi, ""); + if (referencesIndex !== -1 && + alterTableName !== "" && + foreignKeySQL !== "" && + referencesSQL !== "") { + //Remove references syntax + referencesSQL = referencesSQL.replace(/REFERENCES /gi, ""); + //Get Table and Property Index + const referencedTableIndex = referencesSQL.indexOf("("); + const referencedPropertyIndex = referencesSQL.indexOf(")"); + //Get Referenced Table + let referencedTableName = referencesSQL.substring(0, referencedTableIndex); + //Parse Name + referencedTableName = this.ParseSQLServerName(referencedTableName); + //Get Referenced Key + let referencedPropertyName = referencesSQL.substring(referencedTableIndex + 1, referencedPropertyIndex); + //Parse Name + referencedPropertyName = this.ParseSQLServerName(referencedPropertyName); + //Get ForeignKey + let foreignKey = foreignKeySQL + .replace(/FOREIGN KEY\s?\(/gi, "") + .replace(")", ""); + //Parse Name + foreignKey = this.ParseSQLServerName(foreignKey); + //Parse Name + alterTableName = this.ParseSQLServerName(alterTableName); + //Create ForeignKey + const foreignKeyOriginModel = this.CreateForeignKey(foreignKey, alterTableName, referencedPropertyName, referencedTableName, true); + //Add ForeignKey Origin + this.foreignKeyList.push(foreignKeyOriginModel); + //Create ForeignKey + const foreignKeyDestinationModel = this.CreateForeignKey(referencedPropertyName, referencedTableName, foreignKey, alterTableName, false); + //Add ForeignKey Destination + this.foreignKeyList.push(foreignKeyDestinationModel); + } + } + CreateForeignKey(primaryKeyName, primaryKeyTableName, referencesPropertyName, referencesTableName, isDestination) { + const foreignKey = { + PrimaryKeyTableName: this.RemoveNameQuantifiers(primaryKeyTableName), + PrimaryKeyName: this.RemoveNameQuantifiers(primaryKeyName), + ReferencesPropertyName: this.RemoveNameQuantifiers(referencesPropertyName), + ReferencesTableName: this.RemoveNameQuantifiers(referencesTableName), + IsDestination: isDestination !== undefined && isDestination !== null + ? isDestination + : false, + }; + return foreignKey; + } + RemoveNameQuantifiers(name) { + return name.replace(/\[|\]|\(|\"|\'|\`/g, "").trim(); + } + ParseTableName(name) { + if (name.charAt(name.length - 1) === "(") { + name = this.RemoveNameQuantifiers(name); + } + return name; + } + ParseSQLServerName(name, property) { + name = name.replace("[dbo].[", ""); + name = name.replace("](", ""); + name = name.replace("].[", "."); + name = name.replace("[", ""); + if (property == undefined || property == null) { + name = name.replace(" [", ""); + name = name.replace("] ", ""); + } + else { + if (name.indexOf("]") !== -1) { + name = name.substring(0, name.indexOf("]")); + } + } + if (name.lastIndexOf("]") === name.length - 1) { + name = name.substring(0, name.length - 1); + } + if (name.lastIndexOf(")") === name.length - 1) { + name = name.substring(0, name.length - 1); + } + if (name.lastIndexOf("(") === name.length - 1) { + name = name.substring(0, name.length - 1); + } + name = name.replace(" ", ""); + return name; + } + CreateTable(name) { + const table = { + Name: name, + Properties: [], + }; + //Count exported tables + this.exportedTables++; + return table; + } + /** + * Checks whether character is a quotation character. + * + * @param char Character to be evaluated. + */ + static isQuoteChar(char) { + return char === "\"" || char === "'" || char === "`"; + } + /** + * convert labels with start and end strings per database type + * @param label + * @returns + */ + dbTypeEnds(label) { + let char1 = "\""; + let char2 = "\""; + if (this.dialect == "mysql") { + char1 = "`"; + char2 = "`"; + } + else if (this.dialect == "sqlserver") { + char1 = "["; + char2 = "]"; + } + return `${char1}${label}${char2}`; + } + WithEnds() { + this.tableList = this.tableList.map((table) => { + table.Name = this.dbTypeEnds(table.Name); + table.Properties = table.Properties.map((property) => { + property.Name = this.dbTypeEnds(property.Name); + property.TableName = this.dbTypeEnds(property.TableName); + return property; + }); + return table; + }); + this.primaryKeyList = this.primaryKeyList.map((primaryKey) => { + primaryKey.PrimaryKeyName = this.dbTypeEnds(primaryKey.PrimaryKeyName); + primaryKey.PrimaryKeyTableName = this.dbTypeEnds(primaryKey.PrimaryKeyTableName); + return primaryKey; + }); + this.foreignKeyList = this.foreignKeyList.map((foreignKey) => { + foreignKey.PrimaryKeyName = this.dbTypeEnds(foreignKey.PrimaryKeyName); + foreignKey.ReferencesPropertyName = this.dbTypeEnds(foreignKey.ReferencesPropertyName); + foreignKey.PrimaryKeyTableName = this.dbTypeEnds(foreignKey.PrimaryKeyTableName); + foreignKey.ReferencesTableName = this.dbTypeEnds(foreignKey.ReferencesTableName); + return foreignKey; + }); + return this; + } + WithoutEnds() { + this.tableList.map((table) => { + table.Name = this.RemoveNameQuantifiers(table.Name); + table.Properties = table.Properties.map((property) => { + property.Name = this.RemoveNameQuantifiers(property.Name); + property.TableName = this.RemoveNameQuantifiers(property.TableName); + return property; + }); + return table; + }); + this.primaryKeyList = this.primaryKeyList.map((primaryKey) => { + primaryKey.PrimaryKeyName = this.RemoveNameQuantifiers(primaryKey.PrimaryKeyName); + primaryKey.PrimaryKeyTableName = this.RemoveNameQuantifiers(primaryKey.PrimaryKeyTableName); + return primaryKey; + }); + this.foreignKeyList = this.foreignKeyList.map((foreignKey) => { + foreignKey.PrimaryKeyName = this.RemoveNameQuantifiers(foreignKey.PrimaryKeyName); + foreignKey.ReferencesPropertyName = this.RemoveNameQuantifiers(foreignKey.ReferencesPropertyName); + foreignKey.PrimaryKeyTableName = this.RemoveNameQuantifiers(foreignKey.PrimaryKeyTableName); + foreignKey.ReferencesTableName = this.RemoveNameQuantifiers(foreignKey.ReferencesTableName); + return foreignKey; + }); + return this; + } + /** + * return text quantifiers for dialect + * @returns json + */ + GetColumnQuantifiersFromDialect():ColumnQuantifiers { + const chars = { + Start: "\"", + End: "\"", + }; + if (this.dialect == "mysql") { + chars.Start = "`"; + chars.End = "`"; + } + else if (this.dialect == "sqlserver") { + chars.Start = "["; + chars.End = "]"; + } + return chars; + } + ToTableList() { + return this.tableList; + } + ToPrimaryKeyList() { + return this.primaryKeyList; + } + ToForeignKeyList() { + return this.foreignKeyList; + } + ResetModel() { + this.tableList = []; + this.primaryKeyList = []; + this.foreignKeyList = []; + return this; + } + /** + * return full sql model + * @returns + */ + ToModel() { + return { + TableList: this.tableList, + Dialect: this.dialect, + ForeignKeyList: this.foreignKeyList, + PrimaryKeyList: this.primaryKeyList, + }; + } + } + + //Table Info + var foreignKeyList = []; + var primaryKeyList = []; + var tableList = []; + var cells = []; + var tableCell = null; + var rowCell = null; + var dx = 0; + var exportedTables = 0; + + + //Create Base div + const divFromSQL = document.createElement('div'); + divFromSQL.style.userSelect = 'none'; + divFromSQL.style.overflow = 'hidden'; + divFromSQL.style.padding = '10px'; + divFromSQL.style.height = '100%'; + + var graph = ui.editor.graph; + + const sqlInputFromSQL = document.createElement('textarea'); + sqlInputFromSQL.style.height = '200px'; + sqlInputFromSQL.style.width = '100%'; + var defaultReset = '/*\n\tDraw io default value\n*/\n\nCREATE TABLE Persons\n(\n PersonID int NOT NULL,\n LastName varchar(255),\n ' + + 'FirstName varchar(255),\n Address varchar(255),\n City varchar(255),\n Primary Key(PersonID)\n);\n\n' + + 'CREATE TABLE Orders\n(\n OrderID int NOT NULL PRIMARY KEY,\n PersonID int NOT NULL,\n FOREIGN KEY ([PersonID]) REFERENCES [Persons]([PersonID])' + + '\n);' + sqlInputFromSQL.value = defaultReset + mxUtils.br(divFromSQL); + divFromSQL.appendChild(sqlInputFromSQL); + + var graph = ui.editor.graph; + + // Extends Extras menu + mxResources.parse('fromSql=From SQL'); + + const wndFromSQL = new mxWindow(mxResources.get('fromSql'), divFromSQL, document.body.offsetWidth - 480, 140, + 320, 320, true, true); + wndFromSQL.destroyOnClose = false; + wndFromSQL.setMaximizable(false); + wndFromSQL.setResizable(false); + wndFromSQL.setClosable(true); + + function AddRow(propertyModel, tableName) { + + var cellName = propertyModel.Name + (propertyModel.ColumnProperties ? " " + propertyModel.ColumnProperties: ""); + + rowCell = new mxCell(cellName, new mxGeometry(0, 0, 90, 26), + 'shape=partialRectangle;top=0;left=0;right=0;bottom=0;align=left;verticalAlign=top;spacingTop=-2;fillColor=none;spacingLeft=64;spacingRight=4;overflow=hidden;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;dropTarget=0;'); + rowCell.vertex = true; + + var columnType = propertyModel.IsPrimaryKey && propertyModel.IsForeignKey ? 'PK | FK' : propertyModel.IsPrimaryKey ? 'PK' : propertyModel.IsForeignKey ? 'FK' : ''; + + var left = sb.cloneCell(rowCell, columnType); + left.connectable = false; + left.style = 'shape=partialRectangle;top=0;left=0;bottom=0;fillColor=none;align=left;verticalAlign=middle;spacingLeft=4;spacingRight=4;overflow=hidden;rotatable=180;points=[];portConstraint=eastwest;part=1;' + left.geometry.width = 54; + left.geometry.height = 26; + rowCell.insert(left); + + var size = ui.editor.graph.getPreferredSizeForCell(rowCell); + + if (size !== null && tableCell.geometry.width < size.width + 10) { + tableCell.geometry.width = size.width + 10; + } + + tableCell.insert(rowCell); + tableCell.geometry.height += 26; + + rowCell = rowCell; + + }; + + function parseSql(text, type) { + // reset values + cells = [] + tableCell = null; + rowCell = null; + // load parser + const parser = new SqlSimpleParser(type); + + + const models = parser + .feed(text) + .WithoutEnds() + .WithEnds() + .ToModel(); + + + + foreignKeyList = models.ForeignKeyList; + primaryKeyList = models.PrimaryKeyList; + tableList = models.TableList; + exportedTables = tableList.length; + + //Create Table in UI + CreateTableUI(type); + }; + + function CreateTableUI(type) { + tableList.forEach(function(tableModel) { + //Define table size width + var maxNameLenght = 100 + tableModel.Name.length; + + //Create Table + tableCell = new mxCell(tableModel.Name, new mxGeometry(dx, 0, maxNameLenght, 26), + 'swimlane;fontStyle=0;childLayout=stackLayout;horizontal=1;startSize=26;fillColor=default;horizontalStack=0;resizeParent=1;resizeLast=0;collapsible=1;marginBottom=0;swimlaneFillColor=default;align=center;'); + tableCell.vertex = true; + + //Resize row + var size = ui.editor.graph.getPreferredSizeForCell(rowCell); + if (size !== null) { + tableCell.geometry.width = size.width + maxNameLenght; + } + + //Add Table to cells + cells.push(tableCell); + + //Add properties + tableModel.Properties.forEach(function(propertyModel) { + + //Add row + AddRow(propertyModel, tableModel.Name); + }); + + //Close table + dx += tableCell.geometry.width + 40; + tableCell = null; + }); + + if (cells.length > 0) { + var graph = ui.editor.graph; + var view = graph.view; + var bds = graph.getGraphBounds(); + + // Computes unscaled, untranslated graph bounds + var x = Math.ceil(Math.max(0, bds.x / view.scale - view.translate.x) + 4 * graph.gridSize); + var y = Math.ceil(Math.max(0, (bds.y + bds.height) / view.scale - view.translate.y) + 4 * graph.gridSize); + + graph.setSelectionCells(graph.importCells(cells, x, y)); + // add foreign key edges + var model = graph.getModel(); + const columnQuantifiers = GetColumnQuantifiers(type); + var pt = graph.getFreeInsertPoint(); + foreignKeyList.forEach(function(fk){ + if(fk.IsDestination && fk.PrimaryKeyName && fk.ReferencesPropertyName && + fk.PrimaryKeyTableName && fk.ReferencesTableName) { + var insertEdge = mxUtils.bind(this, function(targetCell, sourceCell, edge){ + var label = "" + var edgeStyle = "edgeStyle=entityRelationEdgeStyle;html=1;endArrow=ERzeroToMany;startArrow=ERzeroToOne;labelBackgroundColor=none;fontFamily=Verdana;fontSize=14;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=-0.018;entryY=0.608;entryDx=0;entryDy=0;entryPerimeter=0;" + var edgeCell = graph.insertEdge(null, null, label || '', (edge.invert) ? + sourceCell : targetCell, (edge.invert) ? targetCell : sourceCell, edgeStyle); + }); + let edge = { + invert: true + }; + var targetCell = null; + var sourceCell = null; + // locate edge source and target cells + for (const key in model.cells) { + if(targetCell && sourceCell) + break; + if (Object.hasOwnProperty.call(model.cells, key)) { + const mxcell = model.cells[key]; + if(mxcell.style && mxcell.style.trim().startsWith("swimlane;")){ + let entity = { + name: mxcell.value, + attributes: [] + } + var isPrimaryTable = entity.name == fk.PrimaryKeyTableName; + var isForeignTable = entity.name == fk.ReferencesTableName; + if(isPrimaryTable || isForeignTable){ + for (let c = 0; c < mxcell.children.length; c++) { + if(targetCell && sourceCell) + break; + const col = mxcell.children[c]; + if(col.mxObjectId.indexOf("mxCell") !== -1) { + if(col.style && col.style.trim().startsWith("shape=partialRectangle")){ + let attribute = getDbLabel(col.value, columnQuantifiers) + if(isPrimaryTable && attribute.attributeName == fk.PrimaryKeyName){ + targetCell = col; + break; + } else if(isForeignTable && attribute.attributeName == fk.ReferencesPropertyName){ + sourceCell = col; + break; + } + } + } + } + } + + } + } + } + if(targetCell && sourceCell) + insertEdge(targetCell, sourceCell, edge); + } + }) + graph.scrollCellToVisible(graph.getSelectionCell()); + } + + wndFromSQL.setVisible(false); + }; + + mxUtils.br(divFromSQL); + + const resetBtnFromSQL = mxUtils.button(mxResources.get('reset'), function() { + sqlInputFromSQL.value = defaultReset; + }); + + resetBtnFromSQL.style.marginTop = '8px'; + resetBtnFromSQL.style.marginRight = '4px'; + resetBtnFromSQL.style.padding = '4px'; + divFromSQL.appendChild(resetBtnFromSQL); + + const btnFromSQL_mysql = mxUtils.button('Insert MySQL', function() { + parseSql(sqlInputFromSQL.value, 'mysql'); + }); + + btnFromSQL_mysql.style.marginTop = '8px'; + btnFromSQL_mysql.style.padding = '4px'; + divFromSQL.appendChild(btnFromSQL_mysql); + + const btnFromSQL_sqlserver = mxUtils.button('Insert SQL Server', function() { + parseSql(sqlInputFromSQL.value, 'sqlserver'); + }); + + btnFromSQL_sqlserver.style.marginTop = '8px'; + btnFromSQL_sqlserver.style.padding = '4px'; + divFromSQL.appendChild(btnFromSQL_sqlserver); + + const btnFromSQL_postgres = mxUtils.button('Insert PostgreSQL', function() { + parseSql(sqlInputFromSQL.value, 'postgres'); + }); + + btnFromSQL_postgres.style.marginTop = '8px'; + btnFromSQL_postgres.style.padding = '4px'; + divFromSQL.appendChild(btnFromSQL_postgres); + + const btnFromSQL_sqlite = mxUtils.button('Insert Sqlite', function() { + parseSql(sqlInputFromSQL.value, 'sqlite'); + }); + + btnFromSQL_sqlite.style.marginTop = '8px'; + btnFromSQL_sqlite.style.padding = '4px'; + divFromSQL.appendChild(btnFromSQL_sqlite); + + // Adds action + ui.actions.addAction('fromSql', function() { + wndFromSQL.setVisible(!wndFromSQL.isVisible()); + + if (wndFromSQL.isVisible()) { + sqlInputFromSQL.focus(); + } + }); + // end import diagrams from sql text methods + + // finalize menu buttons + var theMenu = ui.menus.get('insert'); + var oldMenu = theMenu.funct; + + theMenu.funct = function(menu, parent) { + oldMenu.apply(this, arguments); + ui.menus.addMenuItems(menu, ['fromSql'], parent); + }; + if(theMenuExportAs && theMenuExportAs.enabled) { + var oldMenuExportAs = theMenuExportAs.funct; + + theMenuExportAs.funct = function(menu, parent) { + oldMenuExportAs.apply(this, arguments); + ui.menus.addMenuItems(menu, ['tosql'], parent); + }; + } else { + // vscode file export sql menu + var menu = ui.menus.get('file'); + var oldMenuExportAs = menu.funct; + menu.funct = function(menu, parent) { + oldMenuExportAs.apply(this, arguments); + debugger + ui.menus.addMenuItems(menu, ['tosql'], parent); + }; + } +}); \ No newline at end of file diff --git a/src/types/drawio-types.d.ts b/src/types/drawio-types.d.ts new file mode 100644 index 0000000..345b635 --- /dev/null +++ b/src/types/drawio-types.d.ts @@ -0,0 +1,136 @@ +// pulled from https://github.com/hediet/vscode-drawio/blob/master/drawio-custom-plugins/src/drawio-types.d.ts +declare const Draw: { + loadPlugin(handler: (ui: DrawioUI) => void): void; +}; + +declare const log: any; +declare class mxCellHighlight { + constructor(graph: DrawioGraph, color: string, arg: number); + + public highlight(arg: DrawioCellState | null): void; + public destroy(): void; +} + +declare class mxResources { + static parse(value: string): void; + static get(key: string): string; +} + +declare class mxWindow { + /** + * Basic window inside a document. + * @param title String that represents the title of the new window. + * @param content DOM node that is used as the window content. + * @param x X-coordinate of the window location. + * @param y Y-coordinate of the window location. + * @param width Width of the window. + * @param height Optional height of the window. Default is to match the height + * of the content at the specified width. + * @param minimizable Optional boolean indicating if the window is minimizable. + * Default is true. + * @param replaceNode Optional boolean indicating if the window is movable. Default + * is true. + * @param style Optional base classname for the window elements. Default is + * mxWindow. + */ + constructor(title: string, content: HTMLElement, x: number, y: number, width: number, height?: number, minimizable?: boolean, replaceNode?: boolean, style?: string); + public setClosable(value: boolean): void; + public setVisible(value: boolean): void; + public isVisible(): boolean; + public setResizable(value: boolean): void; + public setMaximizable(value: boolean): void; +} + +declare class mxMouseEvent { + public readonly graphX: number; + public readonly graphY: number; +} + +declare const mxEvent: { + DOUBLE_CLICK: string; + CHANGE: string; +}; + +declare const mxUtils: { + button(title: string, funct: () => void): HTMLElement; + isNode(node: any): node is HTMLElement; + createXmlDocument(): XMLDocument; + br(element: HTMLElement); +}; + + +declare interface DrawioUI { + fileNode: Element | null; + hideDialog(): void; + showDialog(...args: any[]): void; + editor: DrawioEditor; + actions: DrawioActions; + menus: DrawioMenus; + importLocalFile(args: boolean): void; +} + +interface DrawioMenus { + get(name: string): any; + addMenuItems(menu: any, arg: any, arg2: any): void; +} + +interface DrawioActions { + addAction(name: string, action: () => void): void; + get(name: string): { funct: () => void }; +} + +declare interface DrawioEditor { + graph: DrawioGraph; +} + +declare interface DrawioGraph { + defaultThemeName: string; + insertVertex(arg0: undefined, arg1: null, label: string, arg3: number, arg4: number, arg5: number, arg6: number, arg7: string): void; + addListener: any; + model: DrawioGraphModel; + getLabel(cell: DrawioCell): string; + getModel(): DrawioGraphModel; + getSelectionModel(): DrawioGraphSelectionModel; + view: DrawioGraphView; + + addMouseListener(listener: { + mouseMove?: (graph: DrawioGraph, event: mxMouseEvent) => void; + mouseDown?: (graph: DrawioGraph, event: mxMouseEvent) => void + mouseUp?: (graph: DrawioGraph, event: mxMouseEvent) => void; + }): void; +} + +declare interface DrawioGraphView { + getState(cell: DrawioCell): DrawioCellState; + canvas: SVGElement; +} + +declare interface DrawioCellState { + cell: DrawioCell; +} + +declare interface DrawioGraphSelectionModel { + addListener(event: string, handler: (...args: any[]) => void): void; + cells: DrawioCell[]; +} + +declare interface DrawioCell { + id: string; + style: string; + value: string; + mxObjectId: string; + children: DrawioCell[]; + edges: DrawioCell[]; + source: DrawioCell; + target?: DrawioCell; + parent: DrawioCell; +} + +declare interface DrawioGraphModel { + setValue(c: DrawioCell, label: string | any): void; + beginUpdate(): void; + endUpdate(): void; + cells: Record; + setStyle(cell: DrawioCell, style: string): void; + isVertex(cell: DrawioCell): boolean; +} \ No newline at end of file diff --git a/src/types/sql-plugin-types.ts b/src/types/sql-plugin-types.ts new file mode 100644 index 0000000..539d6b2 --- /dev/null +++ b/src/types/sql-plugin-types.ts @@ -0,0 +1,17 @@ + +export interface ColumnQuantifiers { + Start: string; + End: string; +} + +export interface TableEntity { + name: string; + attributes: TableAttribute[]; +} + +export interface TableAttribute { + attributeType: string; + attributeName: string; + attributeKeyType?: string; + // "PK" | "FK" +} \ No newline at end of file diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..b9b5312 --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,118 @@ +{ + "compilerOptions": { + /* Visit https://aka.ms/tsconfig to read more about this file */ + + /* Projects */ + // "incremental": true, /* Save .tsbuildinfo files to allow for incremental compilation of projects. */ + // "composite": true, /* Enable constraints that allow a TypeScript project to be used with project references. */ + // "tsBuildInfoFile": "./.tsbuildinfo", /* Specify the path to .tsbuildinfo incremental compilation file. */ + // "disableSourceOfProjectReferenceRedirect": true, /* Disable preferring source files instead of declaration files when referencing composite projects. */ + // "disableSolutionSearching": true, /* Opt a project out of multi-project reference checking when editing. */ + // "disableReferencedProjectLoad": true, /* Reduce the number of projects loaded automatically by TypeScript. */ + + /* Language and Environment */ + "target": "es2016", /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */ + "lib": [ + "DOM", + "ES2021" + ], /* Specify a set of bundled library declaration files that describe the target runtime environment. */ + // "jsx": "preserve", /* Specify what JSX code is generated. */ + // "experimentalDecorators": true, /* Enable experimental support for TC39 stage 2 draft decorators. */ + // "emitDecoratorMetadata": true, /* Emit design-type metadata for decorated declarations in source files. */ + // "jsxFactory": "", /* Specify the JSX factory function used when targeting React JSX emit, e.g. 'React.createElement' or 'h'. */ + // "jsxFragmentFactory": "", /* Specify the JSX Fragment reference used for fragments when targeting React JSX emit e.g. 'React.Fragment' or 'Fragment'. */ + // "jsxImportSource": "", /* Specify module specifier used to import the JSX factory functions when using 'jsx: react-jsx*'. */ + // "reactNamespace": "", /* Specify the object invoked for 'createElement'. This only applies when targeting 'react' JSX emit. */ + // "noLib": true, /* Disable including any library files, including the default lib.d.ts. */ + // "useDefineForClassFields": true, /* Emit ECMAScript-standard-compliant class fields. */ + // "moduleDetection": "auto", /* Control what method is used to detect module-format JS files. */ + + /* Modules */ + "module": "CommonJS", /* Specify what module code is generated. */ + // "rootDir": "./", /* Specify the root folder within your source files. */ + "moduleResolution": "node", /* Specify how TypeScript looks up a file from a given module specifier. */ + // "baseUrl": "./", /* Specify the base directory to resolve non-relative module names. */ + // "paths": {}, /* Specify a set of entries that re-map imports to additional lookup locations. */ + // "rootDirs": [], /* Allow multiple folders to be treated as one when resolving modules. */ + // "typeRoots": [], /* Specify multiple folders that act like './node_modules/@types'. */ + // "types": [], /* Specify type package names to be included without being referenced in a source file. */ + // "allowUmdGlobalAccess": true, /* Allow accessing UMD globals from modules. */ + // "moduleSuffixes": [], /* List of file name suffixes to search when resolving a module. */ + "resolveJsonModule": true, /* Enable importing .json files. */ + // "noResolve": true, /* Disallow 'import's, 'require's or ''s from expanding the number of files TypeScript should add to a project. */ + + /* JavaScript Support */ + "allowJs": true, /* Allow JavaScript files to be a part of your program. Use the 'checkJS' option to get errors from these files. */ + // "checkJs": false, /* Enable error reporting in type-checked JavaScript files. */ + // "maxNodeModuleJsDepth": 1, /* Specify the maximum folder depth used for checking JavaScript files from 'node_modules'. Only applicable with 'allowJs'. */ + + /* Emit */ + // "declaration": true, /* Generate .d.ts files from TypeScript and JavaScript files in your project. */ + // "declarationMap": true, /* Create sourcemaps for d.ts files. */ + // "emitDeclarationOnly": true, /* Only output d.ts files and not JavaScript files. */ + "sourceMap": true, /* Create source map files for emitted JavaScript files. */ + // "outFile": "./", /* Specify a file that bundles all outputs into one JavaScript file. If 'declaration' is true, also designates a file that bundles all .d.ts output. */ + "outDir": "./lib", /* Specify an output folder for all emitted files. */ + // "removeComments": true, /* Disable emitting comments. */ + // "noEmit": true, /* Disable emitting files from a compilation. */ + "importHelpers": false, /* Allow importing helper functions from tslib once per project, instead of including them per-file. */ + // "importsNotUsedAsValues": "remove", /* Specify emit/checking behavior for imports that are only used for types. */ + // "downlevelIteration": true, /* Emit more compliant, but verbose and less performant JavaScript for iteration. */ + // "sourceRoot": "", /* Specify the root path for debuggers to find the reference source code. */ + // "mapRoot": "", /* Specify the location where debugger should locate map files instead of generated locations. */ + // "inlineSourceMap": true, /* Include sourcemap files inside the emitted JavaScript. */ + // "inlineSources": true, /* Include source code in the sourcemaps inside the emitted JavaScript. */ + // "emitBOM": true, /* Emit a UTF-8 Byte Order Mark (BOM) in the beginning of output files. */ + // "newLine": "crlf", /* Set the newline character for emitting files. */ + // "stripInternal": true, /* Disable emitting declarations that have '@internal' in their JSDoc comments. */ + // "noEmitHelpers": true, /* Disable generating custom helper functions like '__extends' in compiled output. */ + // "noEmitOnError": true, /* Disable emitting files if any type checking errors are reported. */ + // "preserveConstEnums": true, /* Disable erasing 'const enum' declarations in generated code. */ + // "declarationDir": "./", /* Specify the output directory for generated declaration files. */ + // "preserveValueImports": true, /* Preserve unused imported values in the JavaScript output that would otherwise be removed. */ + + /* Interop Constraints */ + // "isolatedModules": true, /* Ensure that each file can be safely transpiled without relying on other imports. */ + // "allowSyntheticDefaultImports": true, /* Allow 'import x from y' when a module doesn't have a default export. */ + "esModuleInterop": true, /* Emit additional JavaScript to ease support for importing CommonJS modules. This enables 'allowSyntheticDefaultImports' for type compatibility. */ + // "preserveSymlinks": true, /* Disable resolving symlinks to their realpath. This correlates to the same flag in node. */ + "forceConsistentCasingInFileNames": true, /* Ensure that casing is correct in imports. */ + + /* Type Checking */ + "strict": true, /* Enable all strict type-checking options. */ + // "noImplicitAny": true, /* Enable error reporting for expressions and declarations with an implied 'any' type. */ + // "strictNullChecks": true, /* When type checking, take into account 'null' and 'undefined'. */ + // "strictFunctionTypes": true, /* When assigning functions, check to ensure parameters and the return values are subtype-compatible. */ + // "strictBindCallApply": true, /* Check that the arguments for 'bind', 'call', and 'apply' methods match the original function. */ + // "strictPropertyInitialization": true, /* Check for class properties that are declared but not set in the constructor. */ + // "noImplicitThis": true, /* Enable error reporting when 'this' is given the type 'any'. */ + // "useUnknownInCatchVariables": true, /* Default catch clause variables as 'unknown' instead of 'any'. */ + // "alwaysStrict": true, /* Ensure 'use strict' is always emitted. */ + // "noUnusedLocals": true, /* Enable error reporting when local variables aren't read. */ + // "noUnusedParameters": true, /* Raise an error when a function parameter isn't read. */ + // "exactOptionalPropertyTypes": true, /* Interpret optional property types as written, rather than adding 'undefined'. */ + // "noImplicitReturns": true, /* Enable error reporting for codepaths that do not explicitly return in a function. */ + // "noFallthroughCasesInSwitch": true, /* Enable error reporting for fallthrough cases in switch statements. */ + // "noUncheckedIndexedAccess": true, /* Add 'undefined' to a type when accessed using an index. */ + // "noImplicitOverride": true, /* Ensure overriding members in derived classes are marked with an override modifier. */ + // "noPropertyAccessFromIndexSignature": true, /* Enforces using indexed accessors for keys declared using an indexed type. */ + // "allowUnusedLabels": true, /* Disable error reporting for unused labels. */ + // "allowUnreachableCode": true, /* Disable error reporting for unreachable code. */ + + /* Completeness */ + // "skipDefaultLibCheck": true, /* Skip type checking .d.ts files that are included with TypeScript. */ + "skipLibCheck": true /* Skip type checking all .d.ts files. */ + }, + "include": [ + "src", "./package.json" + // "src", + // "lib" + ], + "exclude": [ + "deps", + "node_modules", + "**/tests/*", + "src/*.spec.ts", + // "src/mermaid" + ] +} From bafa5667168c3f14497205fcb4810d87db51829a Mon Sep 17 00:00:00 2001 From: lastlink Date: Sat, 22 Jun 2024 15:46:59 -0400 Subject: [PATCH 02/12] more cleanup --- .eslintrc | 2 -- .npmignore | 3 +-- 2 files changed, 1 insertion(+), 4 deletions(-) diff --git a/.eslintrc b/.eslintrc index 0490c11..6aadfbf 100644 --- a/.eslintrc +++ b/.eslintrc @@ -30,11 +30,9 @@ "@typescript-eslint/class-name-casing": "off" }, "ignorePatterns": [ - "src/TextHighlighter.js", "lib/*", "deps/*", "dist/*", - "src/mermaid/*", "archived" ], "overrides": [ diff --git a/.npmignore b/.npmignore index 338e802..7366687 100644 --- a/.npmignore +++ b/.npmignore @@ -110,5 +110,4 @@ junit.xml *.bak output-*.sql tests -build -funktechno-little-mermaid-2-the-sql* \ No newline at end of file +build \ No newline at end of file From a3c6031e9c8d32698387238e8d9141eed5c4830d Mon Sep 17 00:00:00 2001 From: lastlink Date: Sat, 22 Jun 2024 21:30:31 -0400 Subject: [PATCH 03/12] no ts errors --- package-lock.json | 24 +- package.json | 3 +- src/sql.ts | 716 +++--------------------------------- src/types/drawio-types.d.ts | 91 ++++- tsconfig.json | 2 +- 5 files changed, 161 insertions(+), 675 deletions(-) diff --git a/package-lock.json b/package-lock.json index eb68d9a..c54886d 100644 --- a/package-lock.json +++ b/package-lock.json @@ -10,7 +10,7 @@ "license": "MIT", "dependencies": { "@funktechno/little-mermaid-2-the-sql": "^0.1.1", - "ts-node": "^9.0.0" + "@funktechno/sqlsimpleparser": "^0.1.0" }, "devDependencies": { "jest": "^29.0.3", @@ -667,6 +667,28 @@ "node": ">=16.x" } }, + "node_modules/@funktechno/sqlsimpleparser": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/@funktechno/sqlsimpleparser/-/sqlsimpleparser-0.1.0.tgz", + "integrity": "sha512-aZ3yxF7/6KxV2bAl4T0fLSD8g6fogxU8xiE8Ht87PFX8mlq/SWRe8FYoLB5aX7cMj0uWkeeORp3SUyN9fcOWAA==", + "dependencies": { + "@types/dompurify": "^2.3.4", + "lodash": "^4.17.21", + "moment-mini": "^2.24.0", + "path": "^0.12.7" + }, + "engines": { + "node": ">=16.x" + } + }, + "node_modules/@funktechno/sqlsimpleparser/node_modules/@types/dompurify": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/@types/dompurify/-/dompurify-2.4.0.tgz", + "integrity": "sha512-IDBwO5IZhrKvHFUl+clZxgf3hn2b/lU6H1KaBShPkQyGJUQ0xwebezIPSuiyGwfz1UzJWQl4M7BDxtHtCCPlTg==", + "dependencies": { + "@types/trusted-types": "*" + } + }, "node_modules/@istanbuljs/load-nyc-config": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/@istanbuljs/load-nyc-config/-/load-nyc-config-1.1.0.tgz", diff --git a/package.json b/package.json index d6c5380..b513d7e 100644 --- a/package.json +++ b/package.json @@ -36,7 +36,8 @@ }, "homepage": "https://github.com/lastlink/sqltooling-drawio#readme", "dependencies": { - "@funktechno/little-mermaid-2-the-sql": "^0.1.1" + "@funktechno/little-mermaid-2-the-sql": "^0.1.1", + "@funktechno/sqlsimpleparser": "^0.1.0" }, "devDependencies": { "jest": "^29.0.3", diff --git a/src/sql.ts b/src/sql.ts index 75f85a2..de6eb9e 100644 --- a/src/sql.ts +++ b/src/sql.ts @@ -1,6 +1,8 @@ import { DbParser } from "@funktechno/little-mermaid-2-the-sql/lib/src/generate-sql-ddl"; import { DbDefinition, DbRelationshipDefinition } from "@funktechno/little-mermaid-2-the-sql/lib/src/types"; import { ColumnQuantifiers, TableAttribute, TableEntity } from "./types/sql-plugin-types"; +import { SqlSimpleParser } from "@funktechno/sqlsimpleparser"; +import { ForeignKeyModel, PrimaryKeyModel, PropertyModel, TableModel } from "@funktechno/sqlsimpleparser/lib/types"; /** * Parse SQL CREATE TABLE. Simple initial version for community to improve. @@ -48,7 +50,7 @@ Draw.loadPlugin(function(ui) { * return text quantifiers for dialect * @returns json */ - function GetColumnQuantifiers(type:string):ColumnQuantifiers { + function GetColumnQuantifiers(type: "mysql" | "sqlserver" | "sqlite" | "postgres" | undefined):ColumnQuantifiers { let chars = { Start: '"', End: '"', @@ -101,7 +103,7 @@ Draw.loadPlugin(function(ui) { return name.replace(/\[|\]|\(|\"|\'|\`/g, "").trim(); } - function getMermaidDiagramDb(type: string):DbDefinition{ + function getMermaidDiagramDb(type: "mysql" | "sqlserver" | "sqlite" | "postgres" | undefined):DbDefinition{ var model = ui.editor.graph.getModel() // same models from mermaid for diagram relationships // only difference is entities is an array rather than object to allow duplicate tables @@ -297,12 +299,12 @@ Draw.loadPlugin(function(ui) { return db; } - function generateSql(type: string) { + function generateSql(type: "mysql" | "sqlserver" | "sqlite" | "postgres" | undefined) { // get diagram model var db = getMermaidDiagramDb(type); // load parser - var parser = new DbParser(type, db) + var parser = new DbParser(type as string, db) // generate sql var sql = parser.getSQLDataDefinition() sql = `/*\n\tGenerated in drawio\n\tDatabase: ${type}\n*/\n\n` + sql @@ -368,650 +370,14 @@ Draw.loadPlugin(function(ui) { // import diagrams from sql text methods - const contants_1 ={} - contants_1.CONSTRAINT_Foreign_Key = contants_1.CONSTRAINT_Primary_Key = contants_1.Foreign_Key = contants_1.Primary_Key = contants_1.CONSTRAINT = contants_1.AlterTable = contants_1.CreateTable = void 0; - contants_1.CreateTable = "create table"; - contants_1.AlterTable = "alter table "; - contants_1.CONSTRAINT = "constraint"; - contants_1.Primary_Key = "primary key"; - contants_1.Foreign_Key = "foreign key"; - contants_1.CONSTRAINT_Primary_Key = "constraint primary key"; - contants_1.CONSTRAINT_Foreign_Key = "constraint foreign key"; - - /** - * Main Parser class - */ - class SqlSimpleParser { - /** - * Parser constructor. - * Default dialect is 'sqlite'. - * Options: "mysql" | "sqlite" | "postgres" | "sqlserver" . - * - * @param dialect SQL dialect ('sqlite'). - */ - constructor(dialect = "sqlite") { - this.tableList = []; - /** - * Parsed statements. - */ - this.statements = []; - /** - * Remains of string feed, after last parsed statement. - */ - this.remains = ""; - /** - * Whether preparser is currently escaped. - */ - this.escaped = false; - /** - * Current quote char of preparser. - */ - this.quoted = ""; - this.exportedTables = 0; - this.SQLServer = "sqlserver"; - this.MODE_SQLSERVER = false; - this.foreignKeyList = []; - this.primaryKeyList = []; - this.dialect = dialect; - this.MODE_SQLSERVER = - this.dialect !== undefined && - this.dialect !== null && - this.dialect == this.SQLServer; - } - /** - * Feed chunk of string into parser. - * - * @param chunk Chunk of string to be parsed. - */ - feed(chunk) { - // - const removedComments = chunk - // remove database comments, multiline, --, and // - .replace(/\/\*[\s\S]*?\*\/|\/\/|--.*/g, "") - .replace(/IF NOT EXISTS/gi, "") - .trim(); - const cleanedLines = removedComments - .split("\n") - // remove empty lines - .filter((n) => n) - // remove multiple spaces - .map((n) => n.replace(/\s+/g, " ").trim()); - // combine lines that are in parenthesis - const lines = []; - let insertSameLine = false; - cleanedLines.forEach((n) => { - if (lines.length > 0){ - if((n[0] == "(" && - lines[lines.length - 1].toLocaleLowerCase().indexOf(contants_1.CreateTable) == - -1) || - insertSameLine) { - if (lines.length > 0) { - insertSameLine = true; - lines[lines.length - 1] += ` ${n}`; - if (n[0] == ")") - insertSameLine = false; - } - } - else if(lines[lines.length - 1].match(/CONSTRAINT/gi) && - (n.match(/FOREIGN KEY/gi) && !n.match(/CONSTRAINT/gi)) - ){ - lines[lines.length - 1] += ` ${n}`; - } - // add to previous line if current has references and previous has foreign key - else if(lines[lines.length - 1].match(/FOREIGN KEY/gi) && - (n.match(/REFERENCES/gi) && !n.match(/FOREIGN KEY/gi)) - ){ - lines[lines.length - 1] += ` ${n}`; - } - else if(n.substring(0,2).toUpperCase() == "ON"){ - lines[lines.length - 1] += ` ${n}`; - } - else { - lines.push(n); - } - } - else { - lines.push(n); - } - }); - let currentTableModel = null; - //Parse SQL to objects - for (let i = 0; i < lines.length; i++) { - // rowCell = null; - const tmp = lines[i].trim(); - const propertyRow = tmp.toLowerCase().trim(); - if (propertyRow[0] == ")") { - // close table - if (currentTableModel) { - this.tableList.push(currentTableModel); - currentTableModel = null; - } - continue; - } - //Parse Table - if (propertyRow.indexOf(contants_1.CreateTable) != -1) { - //Parse row - let name = tmp - .replace(this.stringToRegex(`/${contants_1.CreateTable}/gi`), "") - .trim(); - //Parse Table Name - name = this.ParseTableName(name); - if (currentTableModel !== null) { - //Add table to the list - this.tableList.push(currentTableModel); - } - //Create Table - currentTableModel = this.CreateTable(name); - } - // Parse Properties - else if (tmp !== "(" && - currentTableModel != null && - propertyRow.indexOf(contants_1.AlterTable) == -1) { - //Parse the row - let name = tmp.substring(0, tmp.charAt(tmp.length - 1) === "," ? tmp.length - 1 : tmp.length); - //Attempt to get the Key Type - let propertyType = name.toLowerCase(); - // .substring(0, AlterTable.length).toLowerCase(); - //Add special constraints - if (this.MODE_SQLSERVER) { - if (propertyType.indexOf(contants_1.CONSTRAINT) !== -1 && - propertyType.indexOf(contants_1.Primary_Key) !== -1) { - propertyType = contants_1.CONSTRAINT_Primary_Key; - } - if (propertyType.indexOf(contants_1.CONSTRAINT) !== -1 && - propertyType.indexOf(contants_1.Foreign_Key) !== -1) { - propertyType = contants_1.CONSTRAINT_Foreign_Key; - } - } - //Verify if this is a property that doesn't have a relationship (One minute of silence for the property) - const normalProperty = !propertyType.match(/PRIMARY KEY\s?\(/gi) && - propertyType.indexOf(contants_1.Foreign_Key) == -1 && - propertyType.indexOf(contants_1.CONSTRAINT_Primary_Key) == -1 && - propertyType.indexOf(contants_1.CONSTRAINT_Foreign_Key) == -1; - const nameSkipCheck = name.toUpperCase().trim(); - //Parse properties that don't have relationships - if (normalProperty) { - if (name === "" || name === "" || name === ");") { - continue; - } - let ExtendedProperties = null; - if (this.MODE_SQLSERVER) { - if (nameSkipCheck.indexOf(" ASC") !== -1 || - nameSkipCheck.indexOf(" DESC") !== -1 || - nameSkipCheck.indexOf("EXEC ") !== -1 || - nameSkipCheck.indexOf("WITH ") !== -1 || - nameSkipCheck.indexOf(" ON") !== -1 || - nameSkipCheck.indexOf("ALTER ") !== -1 || - // comments already removed - nameSkipCheck.indexOf("/*") !== -1 || - nameSkipCheck.indexOf(" CONSTRAINT") !== -1 || - nameSkipCheck.indexOf("SET ") !== -1 || - nameSkipCheck.indexOf(" NONCLUSTERED") !== -1 || - // no spaces desired - nameSkipCheck.indexOf("GO") !== -1 || - nameSkipCheck.indexOf("REFERENCES ") !== -1) { - continue; - } - //Get delimiter of column name - const firstSpaceIndex = name[0] == "[" && name.indexOf("]" + " ") !== -1 - ? name.indexOf("]" + " ") - : name.indexOf(" "); - ExtendedProperties = name.substring(firstSpaceIndex + 1).trim(); - //Get full name - name = name.substring(0, firstSpaceIndex); - name = this.RemoveNameQuantifiers(name); - } - else { - const columnQuantifiers = this.GetColumnQuantifiersFromDialect(); - //Get delimiter of column name - const firstSpaceIndex = name[0] == columnQuantifiers.Start && - name.indexOf(columnQuantifiers.End + " ") !== -1 - ? name.indexOf(columnQuantifiers.End + " ") - : name.indexOf(" "); - ExtendedProperties = name.substring(firstSpaceIndex + 1).trim(); - //Get full name - name = name.substring(0, firstSpaceIndex); - name = this.RemoveNameQuantifiers(name); - } - //Create Property - const propertyModel = this.CreateProperty(name, currentTableModel.Name, null, false, ExtendedProperties); - //Add Property to table - currentTableModel.Properties.push(propertyModel); - if (ExtendedProperties.toLocaleLowerCase().indexOf(contants_1.Primary_Key) > -1) { - //Create Primary Key - const primaryKeyModel = this.CreatePrimaryKey(name, currentTableModel.Name); - //Add Primary Key to List - this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); - } - } - else { - //Parse Primary Key - if (propertyType.indexOf(contants_1.Primary_Key) != -1 || - propertyType.indexOf(contants_1.CONSTRAINT_Primary_Key) != -1) { - if (!this.MODE_SQLSERVER) { - const primaryKey = name - .replace(/PRIMARY KEY\s?\(/gi, "") - .replace(")", ""); - //Create Primary Key - const primaryKeyModel = this.CreatePrimaryKey(primaryKey, currentTableModel.Name); - //Add Primary Key to List - this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); - } - else { - if (propertyRow.indexOf(contants_1.Primary_Key) !== -1 && - nameSkipCheck.indexOf("CLUSTERED") === -1) { - const primaryKey = name - .replace(/PRIMARY KEY\s?\(/gi, "") - .replace(")", ""); - //Create Primary Key - const primaryKeyModel = this.CreatePrimaryKey(primaryKey, currentTableModel.Name); - //Add Primary Key to List - this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); - } - else { - const startIndex = name.toLocaleLowerCase().indexOf("("); - const endIndex = name.indexOf(")") + 1; - const primaryKey = name - .substring(startIndex, endIndex) - .replace("(", "") - .replace(")", "") - .replace(/ASC/gi, "") - .trim(); - const columnQuantifiers = this.GetColumnQuantifiersFromDialect(); - //Get delimiter of column name - const firstSpaceIndex = primaryKey[0] == columnQuantifiers.Start && - primaryKey.indexOf(columnQuantifiers.End + " ") !== -1 - ? primaryKey.indexOf(columnQuantifiers.End + " ") - : primaryKey.indexOf(" "); - const primaryKeyRow = firstSpaceIndex == -1 - ? primaryKey - : primaryKey.substring(firstSpaceIndex + 1).trim(); - //Create Primary Key - const primaryKeyModel = this.CreatePrimaryKey(primaryKeyRow, currentTableModel.Name); - //Add Primary Key to List - this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); - } - } - } - } - //Parse Foreign Key - if (propertyType.indexOf(contants_1.Foreign_Key) != -1 || - propertyType.indexOf(contants_1.CONSTRAINT_Foreign_Key) != -1) { - if (!this.MODE_SQLSERVER || propertyRow.indexOf(contants_1.AlterTable) == -1) { - this.ParseMySQLForeignKey(name, currentTableModel); - } - else { - let completeRow = name; - if (nameSkipCheck.indexOf("REFERENCES") === -1) { - const referencesRow = lines[i + 1].trim(); - completeRow = - "ALTER TABLE [dbo].[" + - currentTableModel.Name + - "] WITH CHECK ADD" + - " " + - name + - " " + - referencesRow; - } - this.ParseSQLServerForeignKey(completeRow, currentTableModel); - } - } - } - else if (propertyRow.indexOf(contants_1.AlterTable) != -1) { - if (this.MODE_SQLSERVER) { - //Parse the row - const alterTableRow = tmp.substring(0, tmp.charAt(tmp.length - 1) === "," ? tmp.length - 1 : tmp.length); - const referencesRow = lines[i + 1].trim(); - const completeRow = alterTableRow + " " + referencesRow; - this.ParseSQLServerForeignKey(completeRow, currentTableModel); - } - } - } - // parse fk and primary keys - if (this.primaryKeyList.length > 0) { - this.primaryKeyList.forEach((pk) => { - // find table index - const pkTableIndex = this.tableList.findIndex((t) => t.Name.toLocaleLowerCase() == - pk.PrimaryKeyTableName.toLocaleLowerCase()); - // find property index - if (pkTableIndex > -1) { - const propertyIndex = this.tableList[pkTableIndex].Properties.findIndex((p) => p.Name.toLocaleLowerCase() == - pk.PrimaryKeyName.toLocaleLowerCase()); - if (propertyIndex > -1) { - this.tableList[pkTableIndex].Properties[propertyIndex].IsPrimaryKey = true; - } - } - }); - } - if (this.foreignKeyList.length > 0) { - this.foreignKeyList.forEach((fk) => { - // find table index - const pkTableIndex = this.tableList.findIndex((t) => t.Name.toLocaleLowerCase() == - fk.ReferencesTableName.toLocaleLowerCase()); - // find property index - if (pkTableIndex > -1) { - const propertyIndex = this.tableList[pkTableIndex].Properties.findIndex((p) => p.Name.toLocaleLowerCase() == - fk.PrimaryKeyName.toLocaleLowerCase()); - if (propertyIndex > -1) { - this.tableList[pkTableIndex].Properties[propertyIndex].ForeignKey.push(fk); - if (!fk.IsDestination) { - this.tableList[pkTableIndex].Properties[propertyIndex].IsForeignKey = true; - } - } - } - }); - } - return this; - } - stringToRegex(str) { - // Main regex - const mainResult = str.match(/\/(.+)\/.*/); - const optionsResult = str.match(/\/.+\/(.*)/); - if (mainResult && optionsResult) { - const main = mainResult[1]; - // Regex options - const options = optionsResult[1]; - // Compiled regex - return new RegExp(main, options); - } - return new RegExp("//(.+)/.*/", "//.+/(.*)/"); - } - CreatePrimaryKey(primaryKeyName, primaryKeyTableName) { - const primaryKeyNames = this.RemoveNameQuantifiers(primaryKeyName) - .split(",") - .filter((n) => n) - // remove multiple spaces - .map((n) => n.replace(/\s+/g, " ").trim()); - const primaryKeys = []; - primaryKeyNames.forEach(name => { - const primaryKey = { - PrimaryKeyTableName: primaryKeyTableName, - PrimaryKeyName: name, - }; - primaryKeys.push(primaryKey); - }); - return primaryKeys; - } - CreateProperty(name, tableName, foreignKey, isPrimaryKey, columnProps) { - const isForeignKey = foreignKey !== undefined && foreignKey !== null; - const property = { - Name: name, - ColumnProperties: columnProps, - TableName: tableName, - ForeignKey: foreignKey || [], - IsForeignKey: isForeignKey, - IsPrimaryKey: isPrimaryKey, - }; - return property; - } - ParseMySQLForeignKey(name, currentTableModel) { - const referencesIndex = name.toLowerCase().indexOf("references"); - let foreignKeySQL = name.substring(0, referencesIndex); - foreignKeySQL = foreignKeySQL.substring(foreignKeySQL.toUpperCase().indexOf("FOREIGN KEY")) - let referencesSQL = name.substring(referencesIndex, name.length); - //Remove references syntax - referencesSQL = referencesSQL.replace(/REFERENCES /gi, ""); - //Get Table and Property Index - const referencedTableIndex = referencesSQL.indexOf("("); - const referencedPropertyIndex = referencesSQL.indexOf(")"); - //Get Referenced Table - const referencedTableName = referencesSQL.substring(0, referencedTableIndex); - //Get Referenced Key - const referencedPropertyName = referencesSQL.substring(referencedTableIndex + 1, referencedPropertyIndex); - //Get ForeignKey - const foreignKey = foreignKeySQL - .replace(/FOREIGN KEY\s?\(/gi, "") - .replace(")", "") - .replace(" ", ""); - //Create ForeignKey - const foreignKeyOriginModel = this.CreateForeignKey(foreignKey, currentTableModel.Name, referencedPropertyName, referencedTableName, true); - //Add ForeignKey Origin - this.foreignKeyList.push(foreignKeyOriginModel); - //Create ForeignKey - const foreignKeyDestinationModel = this.CreateForeignKey(referencedPropertyName, referencedTableName, foreignKey, currentTableModel.Name, false); - //Add ForeignKey Destination - this.foreignKeyList.push(foreignKeyDestinationModel); - } - ParseSQLServerForeignKey(name, currentTableModel) { - const referencesIndex = name.toLowerCase().indexOf("references"); - let foreignKeySQL = ""; - if (name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`) !== -1) { - foreignKeySQL = name - .substring(name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`), referencesIndex) - .replace(/FOREIGN KEY\(/gi, "") - .replace(")", ""); - } - else { - foreignKeySQL = name - .substring(name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`), referencesIndex) - .replace(/FOREIGN KEY\s?\(/gi, "") - .replace(")", ""); - } - let referencesSQL = name.substring(referencesIndex, name.length); - const nameSkipCheck = name.toUpperCase().trim(); - let alterTableName = name - .substring(0, nameSkipCheck.indexOf("WITH")) - .replace(/ALTER TABLE /gi, ""); - if (referencesIndex !== -1 && - alterTableName !== "" && - foreignKeySQL !== "" && - referencesSQL !== "") { - //Remove references syntax - referencesSQL = referencesSQL.replace(/REFERENCES /gi, ""); - //Get Table and Property Index - const referencedTableIndex = referencesSQL.indexOf("("); - const referencedPropertyIndex = referencesSQL.indexOf(")"); - //Get Referenced Table - let referencedTableName = referencesSQL.substring(0, referencedTableIndex); - //Parse Name - referencedTableName = this.ParseSQLServerName(referencedTableName); - //Get Referenced Key - let referencedPropertyName = referencesSQL.substring(referencedTableIndex + 1, referencedPropertyIndex); - //Parse Name - referencedPropertyName = this.ParseSQLServerName(referencedPropertyName); - //Get ForeignKey - let foreignKey = foreignKeySQL - .replace(/FOREIGN KEY\s?\(/gi, "") - .replace(")", ""); - //Parse Name - foreignKey = this.ParseSQLServerName(foreignKey); - //Parse Name - alterTableName = this.ParseSQLServerName(alterTableName); - //Create ForeignKey - const foreignKeyOriginModel = this.CreateForeignKey(foreignKey, alterTableName, referencedPropertyName, referencedTableName, true); - //Add ForeignKey Origin - this.foreignKeyList.push(foreignKeyOriginModel); - //Create ForeignKey - const foreignKeyDestinationModel = this.CreateForeignKey(referencedPropertyName, referencedTableName, foreignKey, alterTableName, false); - //Add ForeignKey Destination - this.foreignKeyList.push(foreignKeyDestinationModel); - } - } - CreateForeignKey(primaryKeyName, primaryKeyTableName, referencesPropertyName, referencesTableName, isDestination) { - const foreignKey = { - PrimaryKeyTableName: this.RemoveNameQuantifiers(primaryKeyTableName), - PrimaryKeyName: this.RemoveNameQuantifiers(primaryKeyName), - ReferencesPropertyName: this.RemoveNameQuantifiers(referencesPropertyName), - ReferencesTableName: this.RemoveNameQuantifiers(referencesTableName), - IsDestination: isDestination !== undefined && isDestination !== null - ? isDestination - : false, - }; - return foreignKey; - } - RemoveNameQuantifiers(name) { - return name.replace(/\[|\]|\(|\"|\'|\`/g, "").trim(); - } - ParseTableName(name) { - if (name.charAt(name.length - 1) === "(") { - name = this.RemoveNameQuantifiers(name); - } - return name; - } - ParseSQLServerName(name, property) { - name = name.replace("[dbo].[", ""); - name = name.replace("](", ""); - name = name.replace("].[", "."); - name = name.replace("[", ""); - if (property == undefined || property == null) { - name = name.replace(" [", ""); - name = name.replace("] ", ""); - } - else { - if (name.indexOf("]") !== -1) { - name = name.substring(0, name.indexOf("]")); - } - } - if (name.lastIndexOf("]") === name.length - 1) { - name = name.substring(0, name.length - 1); - } - if (name.lastIndexOf(")") === name.length - 1) { - name = name.substring(0, name.length - 1); - } - if (name.lastIndexOf("(") === name.length - 1) { - name = name.substring(0, name.length - 1); - } - name = name.replace(" ", ""); - return name; - } - CreateTable(name) { - const table = { - Name: name, - Properties: [], - }; - //Count exported tables - this.exportedTables++; - return table; - } - /** - * Checks whether character is a quotation character. - * - * @param char Character to be evaluated. - */ - static isQuoteChar(char) { - return char === "\"" || char === "'" || char === "`"; - } - /** - * convert labels with start and end strings per database type - * @param label - * @returns - */ - dbTypeEnds(label) { - let char1 = "\""; - let char2 = "\""; - if (this.dialect == "mysql") { - char1 = "`"; - char2 = "`"; - } - else if (this.dialect == "sqlserver") { - char1 = "["; - char2 = "]"; - } - return `${char1}${label}${char2}`; - } - WithEnds() { - this.tableList = this.tableList.map((table) => { - table.Name = this.dbTypeEnds(table.Name); - table.Properties = table.Properties.map((property) => { - property.Name = this.dbTypeEnds(property.Name); - property.TableName = this.dbTypeEnds(property.TableName); - return property; - }); - return table; - }); - this.primaryKeyList = this.primaryKeyList.map((primaryKey) => { - primaryKey.PrimaryKeyName = this.dbTypeEnds(primaryKey.PrimaryKeyName); - primaryKey.PrimaryKeyTableName = this.dbTypeEnds(primaryKey.PrimaryKeyTableName); - return primaryKey; - }); - this.foreignKeyList = this.foreignKeyList.map((foreignKey) => { - foreignKey.PrimaryKeyName = this.dbTypeEnds(foreignKey.PrimaryKeyName); - foreignKey.ReferencesPropertyName = this.dbTypeEnds(foreignKey.ReferencesPropertyName); - foreignKey.PrimaryKeyTableName = this.dbTypeEnds(foreignKey.PrimaryKeyTableName); - foreignKey.ReferencesTableName = this.dbTypeEnds(foreignKey.ReferencesTableName); - return foreignKey; - }); - return this; - } - WithoutEnds() { - this.tableList.map((table) => { - table.Name = this.RemoveNameQuantifiers(table.Name); - table.Properties = table.Properties.map((property) => { - property.Name = this.RemoveNameQuantifiers(property.Name); - property.TableName = this.RemoveNameQuantifiers(property.TableName); - return property; - }); - return table; - }); - this.primaryKeyList = this.primaryKeyList.map((primaryKey) => { - primaryKey.PrimaryKeyName = this.RemoveNameQuantifiers(primaryKey.PrimaryKeyName); - primaryKey.PrimaryKeyTableName = this.RemoveNameQuantifiers(primaryKey.PrimaryKeyTableName); - return primaryKey; - }); - this.foreignKeyList = this.foreignKeyList.map((foreignKey) => { - foreignKey.PrimaryKeyName = this.RemoveNameQuantifiers(foreignKey.PrimaryKeyName); - foreignKey.ReferencesPropertyName = this.RemoveNameQuantifiers(foreignKey.ReferencesPropertyName); - foreignKey.PrimaryKeyTableName = this.RemoveNameQuantifiers(foreignKey.PrimaryKeyTableName); - foreignKey.ReferencesTableName = this.RemoveNameQuantifiers(foreignKey.ReferencesTableName); - return foreignKey; - }); - return this; - } - /** - * return text quantifiers for dialect - * @returns json - */ - GetColumnQuantifiersFromDialect():ColumnQuantifiers { - const chars = { - Start: "\"", - End: "\"", - }; - if (this.dialect == "mysql") { - chars.Start = "`"; - chars.End = "`"; - } - else if (this.dialect == "sqlserver") { - chars.Start = "["; - chars.End = "]"; - } - return chars; - } - ToTableList() { - return this.tableList; - } - ToPrimaryKeyList() { - return this.primaryKeyList; - } - ToForeignKeyList() { - return this.foreignKeyList; - } - ResetModel() { - this.tableList = []; - this.primaryKeyList = []; - this.foreignKeyList = []; - return this; - } - /** - * return full sql model - * @returns - */ - ToModel() { - return { - TableList: this.tableList, - Dialect: this.dialect, - ForeignKeyList: this.foreignKeyList, - PrimaryKeyList: this.primaryKeyList, - }; - } - } //Table Info - var foreignKeyList = []; - var primaryKeyList = []; - var tableList = []; - var cells = []; - var tableCell = null; - var rowCell = null; + var foreignKeyList:ForeignKeyModel[] = []; + var primaryKeyList:PrimaryKeyModel[] = []; + var tableList:TableModel[] = []; + var cells:mxCell[] = []; + var tableCell:mxCell|null = null; + var rowCell:mxCell|null = null; var dx = 0; var exportedTables = 0; @@ -1048,7 +414,7 @@ Draw.loadPlugin(function(ui) { wndFromSQL.setResizable(false); wndFromSQL.setClosable(true); - function AddRow(propertyModel, tableName) { + function AddRow(propertyModel:PropertyModel, tableName:string) { var cellName = propertyModel.Name + (propertyModel.ColumnProperties ? " " + propertyModel.ColumnProperties: ""); @@ -1066,19 +432,21 @@ Draw.loadPlugin(function(ui) { rowCell.insert(left); var size = ui.editor.graph.getPreferredSizeForCell(rowCell); + + if(tableCell){ + if (size !== null && tableCell.geometry.width < size.width + 10) { + tableCell.geometry.width = size.width + 10; + } - if (size !== null && tableCell.geometry.width < size.width + 10) { - tableCell.geometry.width = size.width + 10; + tableCell.insert(rowCell); + tableCell.geometry.height += 26; } - tableCell.insert(rowCell); - tableCell.geometry.height += 26; - rowCell = rowCell; }; - function parseSql(text, type) { + function parseSql(text:string, type?: "mysql" | "sqlite" | "postgres" | "sqlserver" | undefined) { // reset values cells = [] tableCell = null; @@ -1104,7 +472,7 @@ Draw.loadPlugin(function(ui) { CreateTableUI(type); }; - function CreateTableUI(type) { + function CreateTableUI(type: "mysql" | "sqlite" | "postgres" | "sqlserver" | undefined) { tableList.forEach(function(tableModel) { //Define table size width var maxNameLenght = 100 + tableModel.Name.length; @@ -1115,9 +483,11 @@ Draw.loadPlugin(function(ui) { tableCell.vertex = true; //Resize row + if(rowCell){ var size = ui.editor.graph.getPreferredSizeForCell(rowCell); - if (size !== null) { - tableCell.geometry.width = size.width + maxNameLenght; + if (size !== null) { + tableCell.geometry.width = size.width + maxNameLenght; + } } //Add Table to cells @@ -1264,27 +634,33 @@ Draw.loadPlugin(function(ui) { // finalize menu buttons var theMenu = ui.menus.get('insert'); - var oldMenu = theMenu.funct; - - theMenu.funct = function(menu, parent) { - oldMenu.apply(this, arguments); - ui.menus.addMenuItems(menu, ['fromSql'], parent); - }; + if(theMenu && theMenu.enabled) { + var oldMenu = theMenu.funct; + theMenu.funct = function(...args) { + const [menu, parent] = args; + oldMenu.apply(this, args); + ui.menus.addMenuItems(menu, ['fromSql'], parent); + }; + } if(theMenuExportAs && theMenuExportAs.enabled) { var oldMenuExportAs = theMenuExportAs.funct; - theMenuExportAs.funct = function(menu, parent) { - oldMenuExportAs.apply(this, arguments); + theMenuExportAs.funct = function(...args) { + const [menu, parent] = args; + oldMenuExportAs.apply(this, args); ui.menus.addMenuItems(menu, ['tosql'], parent); }; } else { // vscode file export sql menu var menu = ui.menus.get('file'); - var oldMenuExportAs = menu.funct; - menu.funct = function(menu, parent) { - oldMenuExportAs.apply(this, arguments); - debugger - ui.menus.addMenuItems(menu, ['tosql'], parent); - }; + if(menu && menu.enabled) { + var oldMenuExportAs = menu.funct; + menu.funct = function(...args) { + const [menu, parent] = args; + oldMenuExportAs.apply(this, args); + debugger + ui.menus.addMenuItems(menu, ['tosql'], parent); + }; + } } }); \ No newline at end of file diff --git a/src/types/drawio-types.d.ts b/src/types/drawio-types.d.ts index 345b635..493267f 100644 --- a/src/types/drawio-types.d.ts +++ b/src/types/drawio-types.d.ts @@ -11,6 +11,71 @@ declare class mxCellHighlight { public destroy(): void; } + +declare class mxCell { + /** + * Cells are the elements of the graph model. They represent the state + * of the groups, vertices and edges in a graph. + * @param value Optional object that represents the cell value. + * @param geometry Optional that specifies the geometry. + * @param style Optional formatted string that defines the style. + */ + constructor(value?:string, geometry?:any, style?: string); + + public vertex: boolean; + public connectable: boolean; + public geometry: mxGeometry; + public style: string; + public insert(arg: mxCell): void; +} + +declare class mxRectangle { + /** + * For vertices, the geometry consists of the x- and y-location, and the width + * and height. For edges, the geometry consists of the optional terminal- and + * control points. The terminal points are only required if an edge is + * unconnected, and are stored in the and + * variables, respectively. + * Extends to represent the geometry of a cell. + * @param x + * @param y + * @param width + * @param height + */ + constructor(x:number, y:number, width:number, height:number); + + public width: number; + public height: number; + public x: number; + public y: number; +} + +declare class mxGeometry extends mxRectangle { + constructor(x:number, y:number, width:number, height:number); +} + + +declare class mxPoint { + /** + * Implements a 2-dimensional vector with double precision coordinates. + * @param x + * @param y + */ + constructor(x:number, y:number); + + public x: number; + public y: number; +} + +declare class sb { + /** + * Adds the general palette to the sidebar. + * @param cell + * @param value + */ + static cloneCell(cell:mxCell, value:string): mxCell; +} + declare class mxResources { static parse(value: string): void; static get(key: string): string; @@ -39,6 +104,7 @@ declare class mxWindow { public isVisible(): boolean; public setResizable(value: boolean): void; public setMaximizable(value: boolean): void; + public destroyOnClose: boolean; } declare class mxMouseEvent { @@ -52,6 +118,7 @@ declare const mxEvent: { }; declare const mxUtils: { + bind(scope: any, funct: (...args: any[]) => void): (...args: any[]) => void; button(title: string, funct: () => void): HTMLElement; isNode(node: any): node is HTMLElement; createXmlDocument(): XMLDocument; @@ -69,8 +136,10 @@ declare interface DrawioUI { importLocalFile(args: boolean): void; } -interface DrawioMenus { - get(name: string): any; +interface DrawioMenus extends Function { + get(name: string): DrawioMenus | null; + funct: (...args: any[]) => void; + enabled: boolean; addMenuItems(menu: any, arg: any, arg2: any): void; } @@ -83,15 +152,31 @@ declare interface DrawioEditor { graph: DrawioGraph; } +interface CellSize { + width: number; + height: number; +} +/** + * graph not mxGeometry + */ declare interface DrawioGraph { defaultThemeName: string; insertVertex(arg0: undefined, arg1: null, label: string, arg3: number, arg4: number, arg5: number, arg6: number, arg7: string): void; addListener: any; model: DrawioGraphModel; + getGraphBounds(): mxRectangle; getLabel(cell: DrawioCell): string; + setSelectionCells(cells: DrawioCell[]); + importCells(cells: mxCell[], x: number, y: number); + getPreferredSizeForCell(cell: mxCell): CellSize; + insertEdge(parent: mxCell | null, id:string | null, value:string, source?: mxCell, target?: mxCell, style?:string): DrawioCell; getModel(): DrawioGraphModel; + getFreeInsertPoint(): mxPoint; + getSelectionCell(): DrawioCell; + scrollCellToVisible(cell: DrawioCell); getSelectionModel(): DrawioGraphSelectionModel; view: DrawioGraphView; + gridSize: number; addMouseListener(listener: { mouseMove?: (graph: DrawioGraph, event: mxMouseEvent) => void; @@ -103,6 +188,8 @@ declare interface DrawioGraph { declare interface DrawioGraphView { getState(cell: DrawioCell): DrawioCellState; canvas: SVGElement; + scale: number; + translate: mxPoint; } declare interface DrawioCellState { diff --git a/tsconfig.json b/tsconfig.json index b9b5312..90ac91a 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -85,7 +85,7 @@ // "strictFunctionTypes": true, /* When assigning functions, check to ensure parameters and the return values are subtype-compatible. */ // "strictBindCallApply": true, /* Check that the arguments for 'bind', 'call', and 'apply' methods match the original function. */ // "strictPropertyInitialization": true, /* Check for class properties that are declared but not set in the constructor. */ - // "noImplicitThis": true, /* Enable error reporting when 'this' is given the type 'any'. */ + "noImplicitThis": false, /* Enable error reporting when 'this' is given the type 'any'. */ // "useUnknownInCatchVariables": true, /* Default catch clause variables as 'unknown' instead of 'any'. */ // "alwaysStrict": true, /* Ensure 'use strict' is always emitted. */ // "noUnusedLocals": true, /* Enable error reporting when local variables aren't read. */ From 63ae99b676705cdf50e736527ca45d214a4f15c6 Mon Sep 17 00:00:00 2001 From: lastlink Date: Sat, 22 Jun 2024 21:38:21 -0400 Subject: [PATCH 04/12] working unit test --- .gitignore | 1 + package-lock.json | 331 ++++++------------------------------------- package.json | 1 + tests/sample.spec.ts | 5 + 4 files changed, 50 insertions(+), 288 deletions(-) create mode 100644 tests/sample.spec.ts diff --git a/.gitignore b/.gitignore index 57d9462..ffbcb58 100644 --- a/.gitignore +++ b/.gitignore @@ -127,3 +127,4 @@ out .yarn/build-state.yml .yarn/install-state.gz .pnp.* +junit.xml diff --git a/package-lock.json b/package-lock.json index c54886d..243a52d 100644 --- a/package-lock.json +++ b/package-lock.json @@ -13,6 +13,7 @@ "@funktechno/sqlsimpleparser": "^0.1.0" }, "devDependencies": { + "@types/jest": "^29.0.3", "jest": "^29.0.3", "jest-junit": "^16.0.0", "jest-serial-runner": "^1.1.0", @@ -819,38 +820,6 @@ "url": "https://github.com/chalk/chalk?sponsor=1" } }, - "node_modules/@jest/core/node_modules/pretty-format": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", - "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", - "dev": true, - "dependencies": { - "@jest/schemas": "^29.6.3", - "ansi-styles": "^5.0.0", - "react-is": "^18.0.0" - }, - "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" - } - }, - "node_modules/@jest/core/node_modules/pretty-format/node_modules/ansi-styles": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", - "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", - "dev": true, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, - "node_modules/@jest/core/node_modules/react-is": { - "version": "18.3.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", - "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", - "dev": true - }, "node_modules/@jest/core/node_modules/strip-ansi": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", @@ -1356,6 +1325,16 @@ "@types/istanbul-lib-report": "*" } }, + "node_modules/@types/jest": { + "version": "29.5.12", + "resolved": "https://registry.npmjs.org/@types/jest/-/jest-29.5.12.tgz", + "integrity": "sha512-eDC8bTvT/QhYdxJAulQikueigY5AsdBRH2yDKW3yveW7svY3+DzN84/2NUgkw10RTiJbWqZrTtoGVdYlvFJdLw==", + "dev": true, + "dependencies": { + "expect": "^29.0.0", + "pretty-format": "^29.0.0" + } + }, "node_modules/@types/mdast": { "version": "3.0.15", "resolved": "https://registry.npmjs.org/@types/mdast/-/mdast-3.0.15.tgz", @@ -3186,38 +3165,6 @@ "url": "https://github.com/chalk/chalk?sponsor=1" } }, - "node_modules/jest-circus/node_modules/pretty-format": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", - "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", - "dev": true, - "dependencies": { - "@jest/schemas": "^29.6.3", - "ansi-styles": "^5.0.0", - "react-is": "^18.0.0" - }, - "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" - } - }, - "node_modules/jest-circus/node_modules/pretty-format/node_modules/ansi-styles": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", - "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", - "dev": true, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, - "node_modules/jest-circus/node_modules/react-is": { - "version": "18.3.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", - "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", - "dev": true - }, "node_modules/jest-cli": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-cli/-/jest-cli-29.7.0.tgz", @@ -3337,38 +3284,6 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, - "node_modules/jest-config/node_modules/pretty-format": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", - "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", - "dev": true, - "dependencies": { - "@jest/schemas": "^29.6.3", - "ansi-styles": "^5.0.0", - "react-is": "^18.0.0" - }, - "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" - } - }, - "node_modules/jest-config/node_modules/pretty-format/node_modules/ansi-styles": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", - "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", - "dev": true, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, - "node_modules/jest-config/node_modules/react-is": { - "version": "18.3.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", - "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", - "dev": true - }, "node_modules/jest-docblock": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-29.7.0.tgz", @@ -3422,38 +3337,6 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, - "node_modules/jest-each/node_modules/pretty-format": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", - "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", - "dev": true, - "dependencies": { - "@jest/schemas": "^29.6.3", - "ansi-styles": "^5.0.0", - "react-is": "^18.0.0" - }, - "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" - } - }, - "node_modules/jest-each/node_modules/pretty-format/node_modules/ansi-styles": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", - "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", - "dev": true, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, - "node_modules/jest-each/node_modules/react-is": { - "version": "18.3.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", - "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", - "dev": true - }, "node_modules/jest-environment-node": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-environment-node/-/jest-environment-node-29.7.0.tgz", @@ -3557,18 +3440,6 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, - "node_modules/jest-leak-detector/node_modules/ansi-styles": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", - "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", - "dev": true, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, "node_modules/jest-leak-detector/node_modules/jest-get-type": { "version": "29.6.3", "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-29.6.3.tgz", @@ -3578,26 +3449,6 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, - "node_modules/jest-leak-detector/node_modules/pretty-format": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", - "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", - "dev": true, - "dependencies": { - "@jest/schemas": "^29.6.3", - "ansi-styles": "^5.0.0", - "react-is": "^18.0.0" - }, - "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" - } - }, - "node_modules/jest-leak-detector/node_modules/react-is": { - "version": "18.3.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", - "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", - "dev": true - }, "node_modules/jest-matcher-utils": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-29.7.0.tgz", @@ -3662,38 +3513,6 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, - "node_modules/jest-matcher-utils/node_modules/pretty-format": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", - "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", - "dev": true, - "dependencies": { - "@jest/schemas": "^29.6.3", - "ansi-styles": "^5.0.0", - "react-is": "^18.0.0" - }, - "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" - } - }, - "node_modules/jest-matcher-utils/node_modules/pretty-format/node_modules/ansi-styles": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", - "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", - "dev": true, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, - "node_modules/jest-matcher-utils/node_modules/react-is": { - "version": "18.3.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", - "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", - "dev": true - }, "node_modules/jest-message-util": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-29.7.0.tgz", @@ -3730,38 +3549,6 @@ "url": "https://github.com/chalk/chalk?sponsor=1" } }, - "node_modules/jest-message-util/node_modules/pretty-format": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", - "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", - "dev": true, - "dependencies": { - "@jest/schemas": "^29.6.3", - "ansi-styles": "^5.0.0", - "react-is": "^18.0.0" - }, - "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" - } - }, - "node_modules/jest-message-util/node_modules/pretty-format/node_modules/ansi-styles": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", - "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", - "dev": true, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, - "node_modules/jest-message-util/node_modules/react-is": { - "version": "18.3.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", - "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", - "dev": true - }, "node_modules/jest-mock": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-mock/-/jest-mock-29.7.0.tgz", @@ -4037,38 +3824,6 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, - "node_modules/jest-snapshot/node_modules/pretty-format": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", - "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", - "dev": true, - "dependencies": { - "@jest/schemas": "^29.6.3", - "ansi-styles": "^5.0.0", - "react-is": "^18.0.0" - }, - "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" - } - }, - "node_modules/jest-snapshot/node_modules/pretty-format/node_modules/ansi-styles": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", - "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", - "dev": true, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, - "node_modules/jest-snapshot/node_modules/react-is": { - "version": "18.3.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", - "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", - "dev": true - }, "node_modules/jest-util": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-29.7.0.tgz", @@ -4156,38 +3911,6 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, - "node_modules/jest-validate/node_modules/pretty-format": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", - "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", - "dev": true, - "dependencies": { - "@jest/schemas": "^29.6.3", - "ansi-styles": "^5.0.0", - "react-is": "^18.0.0" - }, - "engines": { - "node": "^14.15.0 || ^16.10.0 || >=18.0.0" - } - }, - "node_modules/jest-validate/node_modules/pretty-format/node_modules/ansi-styles": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", - "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", - "dev": true, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/chalk/ansi-styles?sponsor=1" - } - }, - "node_modules/jest-validate/node_modules/react-is": { - "version": "18.3.1", - "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", - "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", - "dev": true - }, "node_modules/jest-watcher": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-29.7.0.tgz", @@ -5276,6 +4999,32 @@ "node": ">=8" } }, + "node_modules/pretty-format": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", + "integrity": "sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==", + "dev": true, + "dependencies": { + "@jest/schemas": "^29.6.3", + "ansi-styles": "^5.0.0", + "react-is": "^18.0.0" + }, + "engines": { + "node": "^14.15.0 || ^16.10.0 || >=18.0.0" + } + }, + "node_modules/pretty-format/node_modules/ansi-styles": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", + "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, "node_modules/process": { "version": "0.11.10", "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz", @@ -5322,6 +5071,12 @@ } ] }, + "node_modules/react-is": { + "version": "18.3.1", + "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", + "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", + "dev": true + }, "node_modules/require-directory": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", diff --git a/package.json b/package.json index b513d7e..39a4897 100644 --- a/package.json +++ b/package.json @@ -40,6 +40,7 @@ "@funktechno/sqlsimpleparser": "^0.1.0" }, "devDependencies": { + "@types/jest": "^29.0.3", "jest": "^29.0.3", "jest-junit": "^16.0.0", "jest-serial-runner": "^1.1.0", diff --git a/tests/sample.spec.ts b/tests/sample.spec.ts new file mode 100644 index 0000000..930a6d0 --- /dev/null +++ b/tests/sample.spec.ts @@ -0,0 +1,5 @@ +describe("Application down", () => { + it("Sample", () => { + expect(1).toBeTruthy(); + }); +}); From e86c1325bfe4f2cc81596859013f76cb4f41a82f Mon Sep 17 00:00:00 2001 From: lastlink Date: Sat, 22 Jun 2024 21:48:07 -0400 Subject: [PATCH 05/12] lint fix --- package-lock.json | 3018 +++++++++++++++++++++++++++++++++-- package.json | 10 +- src/sql.ts | 392 +++-- src/types/drawio-types.d.ts | 14 +- 4 files changed, 3085 insertions(+), 349 deletions(-) diff --git a/package-lock.json b/package-lock.json index 243a52d..5c07228 100644 --- a/package-lock.json +++ b/package-lock.json @@ -10,10 +10,16 @@ "license": "MIT", "dependencies": { "@funktechno/little-mermaid-2-the-sql": "^0.1.1", - "@funktechno/sqlsimpleparser": "^0.1.0" + "@funktechno/sqlsimpleparser": "^0.1.0", + "browserify": "^17.0.0", + "tsify": "^5.0.2" }, "devDependencies": { + "@types/eslint": "^7.2.4", "@types/jest": "^29.0.3", + "@typescript-eslint/eslint-plugin": "^2.24.0", + "@typescript-eslint/parser": "^2.24.0", + "eslint": "^6.8.0", "jest": "^29.0.3", "jest-junit": "^16.0.0", "jest-serial-runner": "^1.1.0", @@ -1292,6 +1298,28 @@ "@types/trusted-types": "*" } }, + "node_modules/@types/eslint": { + "version": "7.29.0", + "resolved": "https://registry.npmjs.org/@types/eslint/-/eslint-7.29.0.tgz", + "integrity": "sha512-VNcvioYDH8/FxaeTKkM4/TiTwt6pBV9E3OfGmvaw8tPl0rrHCJ4Ll15HRT+pMiFAf/MLQvAzC+6RzUMEL9Ceng==", + "dev": true, + "dependencies": { + "@types/estree": "*", + "@types/json-schema": "*" + } + }, + "node_modules/@types/eslint-visitor-keys": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/@types/eslint-visitor-keys/-/eslint-visitor-keys-1.0.0.tgz", + "integrity": "sha512-OCutwjDZ4aFS6PB1UZ988C4YgwlBHJd6wCeQqaLdmadZ/7e+w79+hbMUFC1QXDNCmdyoRfAFdm0RypzwR+Qpag==", + "dev": true + }, + "node_modules/@types/estree": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.5.tgz", + "integrity": "sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==", + "dev": true + }, "node_modules/@types/graceful-fs": { "version": "4.1.9", "resolved": "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.9.tgz", @@ -1335,6 +1363,12 @@ "pretty-format": "^29.0.0" } }, + "node_modules/@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 + }, "node_modules/@types/mdast": { "version": "3.0.15", "resolved": "https://registry.npmjs.org/@types/mdast/-/mdast-3.0.15.tgz", @@ -1385,6 +1419,191 @@ "integrity": "sha512-I4q9QU9MQv4oEOz4tAHJtNz1cwuLxn2F3xcc2iV5WdqLPpUnj30aUuxt1mAxYTG+oe8CZMV/+6rU4S4gRDzqtQ==", "dev": true }, + "node_modules/@typescript-eslint/eslint-plugin": { + "version": "2.34.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-2.34.0.tgz", + "integrity": "sha512-4zY3Z88rEE99+CNvTbXSyovv2z9PNOVffTWD2W8QF5s2prBQtwN2zadqERcrHpcR7O/+KMI3fcTAmUUhK/iQcQ==", + "dev": true, + "dependencies": { + "@typescript-eslint/experimental-utils": "2.34.0", + "functional-red-black-tree": "^1.0.1", + "regexpp": "^3.0.0", + "tsutils": "^3.17.1" + }, + "engines": { + "node": "^8.10.0 || ^10.13.0 || >=11.10.1" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "@typescript-eslint/parser": "^2.0.0", + "eslint": "^5.0.0 || ^6.0.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/eslint-plugin/node_modules/regexpp": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.2.0.tgz", + "integrity": "sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg==", + "dev": true, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/mysticatea" + } + }, + "node_modules/@typescript-eslint/experimental-utils": { + "version": "2.34.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/experimental-utils/-/experimental-utils-2.34.0.tgz", + "integrity": "sha512-eS6FTkq+wuMJ+sgtuNTtcqavWXqsflWcfBnlYhg/nS4aZ1leewkXGbvBhaapn1q6qf4M71bsR1tez5JTRMuqwA==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.3", + "@typescript-eslint/typescript-estree": "2.34.0", + "eslint-scope": "^5.0.0", + "eslint-utils": "^2.0.0" + }, + "engines": { + "node": "^8.10.0 || ^10.13.0 || >=11.10.1" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "*" + } + }, + "node_modules/@typescript-eslint/experimental-utils/node_modules/eslint-utils": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.1.0.tgz", + "integrity": "sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg==", + "dev": true, + "dependencies": { + "eslint-visitor-keys": "^1.1.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/mysticatea" + } + }, + "node_modules/@typescript-eslint/parser": { + "version": "2.34.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-2.34.0.tgz", + "integrity": "sha512-03ilO0ucSD0EPTw2X4PntSIRFtDPWjrVq7C3/Z3VQHRC7+13YB55rcJI3Jt+YgeHbjUdJPcPa7b23rXCBokuyA==", + "dev": true, + "dependencies": { + "@types/eslint-visitor-keys": "^1.0.0", + "@typescript-eslint/experimental-utils": "2.34.0", + "@typescript-eslint/typescript-estree": "2.34.0", + "eslint-visitor-keys": "^1.1.0" + }, + "engines": { + "node": "^8.10.0 || ^10.13.0 || >=11.10.1" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^5.0.0 || ^6.0.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/typescript-estree": { + "version": "2.34.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-2.34.0.tgz", + "integrity": "sha512-OMAr+nJWKdlVM9LOqCqh3pQQPwxHAN7Du8DR6dmwCrAmxtiXQnhHJ6tBNtf+cggqfo51SG/FCwnKhXCIM7hnVg==", + "dev": true, + "dependencies": { + "debug": "^4.1.1", + "eslint-visitor-keys": "^1.1.0", + "glob": "^7.1.6", + "is-glob": "^4.0.1", + "lodash": "^4.17.15", + "semver": "^7.3.2", + "tsutils": "^3.17.1" + }, + "engines": { + "node": "^8.10.0 || ^10.13.0 || >=11.10.1" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/acorn": { + "version": "7.4.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.1.tgz", + "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", + "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": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/acorn-node": { + "version": "1.8.2", + "resolved": "https://registry.npmjs.org/acorn-node/-/acorn-node-1.8.2.tgz", + "integrity": "sha512-8mt+fslDufLYntIoPAaIMUe/lrbrehIiwmR3t2k9LljIzoigEPF27eLk2hy8zSGzmR/ogr7zbRKINMo1u0yh5A==", + "dependencies": { + "acorn": "^7.0.0", + "acorn-walk": "^7.0.0", + "xtend": "^4.0.2" + } + }, + "node_modules/acorn-walk": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-7.2.0.tgz", + "integrity": "sha512-OPdCF6GsMIP+Az+aWfAAOEt2/+iVDKE7oy6lJ098aoe59oAmK76qV6Gw60SbZ8jHuG2wH058GF4pLFbYamYrVA==", + "engines": { + "node": ">=0.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==", + "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" + } + }, "node_modules/amdefine": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/amdefine/-/amdefine-1.0.1.tgz", @@ -1445,6 +1664,11 @@ "url": "https://github.com/chalk/ansi-styles?sponsor=1" } }, + "node_modules/any-promise": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/any-promise/-/any-promise-1.3.0.tgz", + "integrity": "sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==" + }, "node_modules/anymatch": { "version": "3.1.3", "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", @@ -1473,6 +1697,53 @@ "sprintf-js": "~1.0.2" } }, + "node_modules/asn1.js": { + "version": "4.10.1", + "resolved": "https://registry.npmjs.org/asn1.js/-/asn1.js-4.10.1.tgz", + "integrity": "sha512-p32cOF5q0Zqs9uBiONKYLm6BClCoBCM5O9JfeUSlnQLBTxYdTK+pW+nXflm8UkKd2UYlEbYz5qEi0JuZR9ckSw==", + "dependencies": { + "bn.js": "^4.0.0", + "inherits": "^2.0.1", + "minimalistic-assert": "^1.0.0" + } + }, + "node_modules/asn1.js/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==" + }, + "node_modules/assert": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/assert/-/assert-1.5.1.tgz", + "integrity": "sha512-zzw1uCAgLbsKwBfFc8CX78DDg+xZeBksSO3vwVIDDN5i94eOrPsSSyiVhmsSABFDM/OcpE2aagCat9dnWQLG1A==", + "dependencies": { + "object.assign": "^4.1.4", + "util": "^0.10.4" + } + }, + "node_modules/astral-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-1.0.0.tgz", + "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "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==", + "dependencies": { + "possible-typed-array-names": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/babel-jest": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/babel-jest/-/babel-jest-29.7.0.tgz", @@ -1608,14 +1879,36 @@ "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==", - "dev": true + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" + }, + "node_modules/base64-js": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", + "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/bn.js": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-5.2.1.tgz", + "integrity": "sha512-eXRvHzWyYPBuB4NBy0cmYQjGitUrtqwbvlzP3G6VFnNRbsZQIxQ10PbKKHt8gZ/HW/D/747aDl+QkDqg3KQLMQ==" }, "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==", - "dev": true, "dependencies": { "balanced-match": "^1.0.0", "concat-map": "0.0.1" @@ -1633,6 +1926,184 @@ "node": ">=8" } }, + "node_modules/brorand": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz", + "integrity": "sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w==" + }, + "node_modules/browser-pack": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/browser-pack/-/browser-pack-6.1.0.tgz", + "integrity": "sha512-erYug8XoqzU3IfcU8fUgyHqyOXqIE4tUTTQ+7mqUjQlvnXkOO6OlT9c/ZoJVHYoAaqGxr09CN53G7XIsO4KtWA==", + "dependencies": { + "combine-source-map": "~0.8.0", + "defined": "^1.0.0", + "JSONStream": "^1.0.3", + "safe-buffer": "^5.1.1", + "through2": "^2.0.0", + "umd": "^3.0.0" + }, + "bin": { + "browser-pack": "bin/cmd.js" + } + }, + "node_modules/browser-resolve": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/browser-resolve/-/browser-resolve-2.0.0.tgz", + "integrity": "sha512-7sWsQlYL2rGLy2IWm8WL8DCTJvYLc/qlOnsakDac87SOoCd16WLsaAMdCiAqsTNHIe+SXfaqyxyo6THoWqs8WQ==", + "dependencies": { + "resolve": "^1.17.0" + } + }, + "node_modules/browserify": { + "version": "17.0.0", + "resolved": "https://registry.npmjs.org/browserify/-/browserify-17.0.0.tgz", + "integrity": "sha512-SaHqzhku9v/j6XsQMRxPyBrSP3gnwmE27gLJYZgMT2GeK3J0+0toN+MnuNYDfHwVGQfLiMZ7KSNSIXHemy905w==", + "dependencies": { + "assert": "^1.4.0", + "browser-pack": "^6.0.1", + "browser-resolve": "^2.0.0", + "browserify-zlib": "~0.2.0", + "buffer": "~5.2.1", + "cached-path-relative": "^1.0.0", + "concat-stream": "^1.6.0", + "console-browserify": "^1.1.0", + "constants-browserify": "~1.0.0", + "crypto-browserify": "^3.0.0", + "defined": "^1.0.0", + "deps-sort": "^2.0.1", + "domain-browser": "^1.2.0", + "duplexer2": "~0.1.2", + "events": "^3.0.0", + "glob": "^7.1.0", + "has": "^1.0.0", + "htmlescape": "^1.1.0", + "https-browserify": "^1.0.0", + "inherits": "~2.0.1", + "insert-module-globals": "^7.2.1", + "JSONStream": "^1.0.3", + "labeled-stream-splicer": "^2.0.0", + "mkdirp-classic": "^0.5.2", + "module-deps": "^6.2.3", + "os-browserify": "~0.3.0", + "parents": "^1.0.1", + "path-browserify": "^1.0.0", + "process": "~0.11.0", + "punycode": "^1.3.2", + "querystring-es3": "~0.2.0", + "read-only-stream": "^2.0.0", + "readable-stream": "^2.0.2", + "resolve": "^1.1.4", + "shasum-object": "^1.0.0", + "shell-quote": "^1.6.1", + "stream-browserify": "^3.0.0", + "stream-http": "^3.0.0", + "string_decoder": "^1.1.1", + "subarg": "^1.0.0", + "syntax-error": "^1.1.1", + "through2": "^2.0.0", + "timers-browserify": "^1.0.1", + "tty-browserify": "0.0.1", + "url": "~0.11.0", + "util": "~0.12.0", + "vm-browserify": "^1.0.0", + "xtend": "^4.0.0" + }, + "bin": { + "browserify": "bin/cmd.js" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/browserify-aes": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/browserify-aes/-/browserify-aes-1.2.0.tgz", + "integrity": "sha512-+7CHXqGuspUn/Sl5aO7Ea0xWGAtETPXNSAjHo48JfLdPWcMng33Xe4znFvQweqc/uzk5zSOI3H52CYnjCfb5hA==", + "dependencies": { + "buffer-xor": "^1.0.3", + "cipher-base": "^1.0.0", + "create-hash": "^1.1.0", + "evp_bytestokey": "^1.0.3", + "inherits": "^2.0.1", + "safe-buffer": "^5.0.1" + } + }, + "node_modules/browserify-cipher": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/browserify-cipher/-/browserify-cipher-1.0.1.tgz", + "integrity": "sha512-sPhkz0ARKbf4rRQt2hTpAHqn47X3llLkUGn+xEJzLjwY8LRs2p0v7ljvI5EyoRO/mexrNunNECisZs+gw2zz1w==", + "dependencies": { + "browserify-aes": "^1.0.4", + "browserify-des": "^1.0.0", + "evp_bytestokey": "^1.0.0" + } + }, + "node_modules/browserify-des": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/browserify-des/-/browserify-des-1.0.2.tgz", + "integrity": "sha512-BioO1xf3hFwz4kc6iBhI3ieDFompMhrMlnDFC4/0/vd5MokpuAc3R+LYbwTA9A5Yc9pq9UYPqffKpW2ObuwX5A==", + "dependencies": { + "cipher-base": "^1.0.1", + "des.js": "^1.0.0", + "inherits": "^2.0.1", + "safe-buffer": "^5.1.2" + } + }, + "node_modules/browserify-rsa": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/browserify-rsa/-/browserify-rsa-4.1.0.tgz", + "integrity": "sha512-AdEER0Hkspgno2aR97SAf6vi0y0k8NuOpGnVH3O99rcA5Q6sh8QxcngtHuJ6uXwnfAXNM4Gn1Gb7/MV1+Ymbog==", + "dependencies": { + "bn.js": "^5.0.0", + "randombytes": "^2.0.1" + } + }, + "node_modules/browserify-sign": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/browserify-sign/-/browserify-sign-4.2.3.tgz", + "integrity": "sha512-JWCZW6SKhfhjJxO8Tyiiy+XYB7cqd2S5/+WeYHsKdNKFlCBhKbblba1A/HN/90YwtxKc8tCErjffZl++UNmGiw==", + "dependencies": { + "bn.js": "^5.2.1", + "browserify-rsa": "^4.1.0", + "create-hash": "^1.2.0", + "create-hmac": "^1.1.7", + "elliptic": "^6.5.5", + "hash-base": "~3.0", + "inherits": "^2.0.4", + "parse-asn1": "^5.1.7", + "readable-stream": "^2.3.8", + "safe-buffer": "^5.2.1" + }, + "engines": { + "node": ">= 0.12" + } + }, + "node_modules/browserify-sign/node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, + "node_modules/browserify-zlib": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/browserify-zlib/-/browserify-zlib-0.2.0.tgz", + "integrity": "sha512-Z942RysHXmJrhqk88FmKBVq/v5tqmSkDz7p54G/MGyjMnCFFnC79XWNbg+Vta8W6Wb2qtSZTSxIGkJrRpCFEiA==", + "dependencies": { + "pako": "~1.0.5" + } + }, + "node_modules/browserify/node_modules/util": { + "version": "0.12.5", + "resolved": "https://registry.npmjs.org/util/-/util-0.12.5.tgz", + "integrity": "sha512-kZf/K6hEIrWHI6XqOFUiiMa+79wE/D8Q+NCNAWclkyg3b4d2k7s0QGepNjiABc+aR3N1PAyHL7p6UcLY6LmrnA==", + "dependencies": { + "inherits": "^2.0.3", + "is-arguments": "^1.0.4", + "is-generator-function": "^1.0.7", + "is-typed-array": "^1.1.3", + "which-typed-array": "^1.1.2" + } + }, "node_modules/browserslist": { "version": "4.23.1", "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.23.1.tgz", @@ -1686,11 +2157,52 @@ "node-int64": "^0.4.0" } }, + "node_modules/buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.2.1.tgz", + "integrity": "sha512-c+Ko0loDaFfuPWiL02ls9Xd3GO3cPVmUobQ6t3rXNUk304u6hGq+8N/kFi+QEIKhzK3uwolVhLzszmfLmMLnqg==", + "dependencies": { + "base64-js": "^1.0.2", + "ieee754": "^1.1.4" + } + }, "node_modules/buffer-from": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", - "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", - "dev": true + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==" + }, + "node_modules/buffer-xor": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/buffer-xor/-/buffer-xor-1.0.3.tgz", + "integrity": "sha512-571s0T7nZWK6vB67HI5dyUF7wXiNcfaPPPTl6zYCNApANjIvYJTg7hlud/+cJpdAhS7dVzqMLmfhfHR3rAcOjQ==" + }, + "node_modules/builtin-status-codes": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/builtin-status-codes/-/builtin-status-codes-3.0.0.tgz", + "integrity": "sha512-HpGFw18DgFWlncDfjTa2rcQ4W88O1mC8e8yZ2AvQY5KDaktSTwo+KRf6nHK6FRI5FyRyb/5T6+TSxfP7QyGsmQ==" + }, + "node_modules/cached-path-relative": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/cached-path-relative/-/cached-path-relative-1.1.0.tgz", + "integrity": "sha512-WF0LihfemtesFcJgO7xfOoOcnWzY/QHR4qeDqV44jPU3HTI54+LnfXK3SA27AVVGCdZFgjjFFaqUA9Jx7dMJZA==" + }, + "node_modules/call-bind": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.7.tgz", + "integrity": "sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==", + "dependencies": { + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.4", + "set-function-length": "^1.2.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } }, "node_modules/callsites": { "version": "3.1.0", @@ -1759,6 +2271,12 @@ "url": "https://github.com/sponsors/wooorm" } }, + "node_modules/chardet": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz", + "integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==", + "dev": true + }, "node_modules/ci-info": { "version": "3.9.0", "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.9.0.tgz", @@ -1774,6 +2292,15 @@ "node": ">=8" } }, + "node_modules/cipher-base": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/cipher-base/-/cipher-base-1.0.4.tgz", + "integrity": "sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q==", + "dependencies": { + "inherits": "^2.0.1", + "safe-buffer": "^5.0.1" + } + }, "node_modules/cjs-module-lexer": { "version": "1.3.1", "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-1.3.1.tgz", @@ -1799,6 +2326,27 @@ "node": "*" } }, + "node_modules/cli-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz", + "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", + "dev": true, + "dependencies": { + "restore-cursor": "^3.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cli-width": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-3.0.0.tgz", + "integrity": "sha512-FxqpkPPwu1HjuN93Omfm4h8uIanXofW0RxVEW3k5RKx+mJJYSthzNhp32Kzxxy3YAEZ/Dc/EWN1vZRY0+kOhbw==", + "dev": true, + "engines": { + "node": ">= 10" + } + }, "node_modules/cliui": { "version": "8.0.1", "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", @@ -1867,6 +2415,25 @@ "node": ">=0.1.90" } }, + "node_modules/combine-source-map": { + "version": "0.8.0", + "resolved": "https://registry.npmjs.org/combine-source-map/-/combine-source-map-0.8.0.tgz", + "integrity": "sha512-UlxQ9Vw0b/Bt/KYwCFqdEwsQ1eL8d1gibiFb7lxQJFdvTgc2hIZi6ugsg+kyhzhPV+QEpUiEIwInIAIrgoEkrg==", + "dependencies": { + "convert-source-map": "~1.1.0", + "inline-source-map": "~0.6.0", + "lodash.memoize": "~3.0.3", + "source-map": "~0.5.3" + } + }, + "node_modules/combine-source-map/node_modules/source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha512-LbrmJOMUSdEVxIKvdcJzQC+nQhe8FUZQTXQy6+I75skNgn3OoQ0DZA8YnFa7gp8tqtL3KPf1kmo0R5DoApeSGQ==", + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/commander": { "version": "11.1.0", "resolved": "https://registry.npmjs.org/commander/-/commander-11.1.0.tgz", @@ -1878,20 +2445,92 @@ "node_modules/concat-map": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", - "dev": true - }, - "node_modules/cose-base": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/cose-base/-/cose-base-1.0.3.tgz", - "integrity": "sha512-s9whTXInMSgAp/NVXVNuVxVKzGH2qck3aQlVHxDCdAEPgtMKwc4Wq6/QKhgdEdgbLSi9rBTAcPoRa6JpiG4ksg==", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==" + }, + "node_modules/concat-stream": { + "version": "1.6.2", + "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz", + "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", + "engines": [ + "node >= 0.8" + ], "dependencies": { - "layout-base": "^1.0.0" + "buffer-from": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^2.2.2", + "typedarray": "^0.0.6" } }, - "node_modules/create-jest": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/create-jest/-/create-jest-29.7.0.tgz", + "node_modules/console-browserify": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/console-browserify/-/console-browserify-1.2.0.tgz", + "integrity": "sha512-ZMkYO/LkF17QvCPqM0gxw8yUzigAOZOSWSHg91FH6orS7vcEj5dVZTidN2fQ14yBSdg97RqhSNwLUXInd52OTA==" + }, + "node_modules/constants-browserify": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/constants-browserify/-/constants-browserify-1.0.0.tgz", + "integrity": "sha512-xFxOwqIzR/e1k1gLiWEophSCMqXcwVHIH7akf7b/vxcUeGunlj3hvZaaqxwHsTgn+IndtkQJgSztIDWeumWJDQ==" + }, + "node_modules/convert-source-map": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.1.3.tgz", + "integrity": "sha512-Y8L5rp6jo+g9VEPgvqNfEopjTR4OTYct8lXlS8iVQdmnjDvbdbzYe9rjtFCB9egC86JoNCU61WRY+ScjkZpnIg==" + }, + "node_modules/core-util-is": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz", + "integrity": "sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==" + }, + "node_modules/cose-base": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/cose-base/-/cose-base-1.0.3.tgz", + "integrity": "sha512-s9whTXInMSgAp/NVXVNuVxVKzGH2qck3aQlVHxDCdAEPgtMKwc4Wq6/QKhgdEdgbLSi9rBTAcPoRa6JpiG4ksg==", + "dependencies": { + "layout-base": "^1.0.0" + } + }, + "node_modules/create-ecdh": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/create-ecdh/-/create-ecdh-4.0.4.tgz", + "integrity": "sha512-mf+TCx8wWc9VpuxfP2ht0iSISLZnt0JgWlrOKZiNqyUZWnjIaCIVNQArMHnCZKfEYRg6IM7A+NeJoN8gf/Ws0A==", + "dependencies": { + "bn.js": "^4.1.0", + "elliptic": "^6.5.3" + } + }, + "node_modules/create-ecdh/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==" + }, + "node_modules/create-hash": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/create-hash/-/create-hash-1.2.0.tgz", + "integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==", + "dependencies": { + "cipher-base": "^1.0.1", + "inherits": "^2.0.1", + "md5.js": "^1.3.4", + "ripemd160": "^2.0.1", + "sha.js": "^2.4.0" + } + }, + "node_modules/create-hmac": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/create-hmac/-/create-hmac-1.1.7.tgz", + "integrity": "sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==", + "dependencies": { + "cipher-base": "^1.0.3", + "create-hash": "^1.1.0", + "inherits": "^2.0.1", + "ripemd160": "^2.0.0", + "safe-buffer": "^5.0.1", + "sha.js": "^2.4.8" + } + }, + "node_modules/create-jest": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/create-jest/-/create-jest-29.7.0.tgz", "integrity": "sha512-Adz2bdH0Vq3F53KEMJOoftQFutWCukm6J24wbPWRO4k1kMY7gS7ds/uoJkNuV8wDCtWWnuwGcJwpWcih+zEW1Q==", "dev": true, "dependencies": { @@ -1932,6 +2571,52 @@ "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==", "dev": true }, + "node_modules/cross-spawn": { + "version": "6.0.5", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", + "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", + "dev": true, + "dependencies": { + "nice-try": "^1.0.4", + "path-key": "^2.0.1", + "semver": "^5.5.0", + "shebang-command": "^1.2.0", + "which": "^1.2.9" + }, + "engines": { + "node": ">=4.8" + } + }, + "node_modules/cross-spawn/node_modules/semver": { + "version": "5.7.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.2.tgz", + "integrity": "sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==", + "dev": true, + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/crypto-browserify": { + "version": "3.12.0", + "resolved": "https://registry.npmjs.org/crypto-browserify/-/crypto-browserify-3.12.0.tgz", + "integrity": "sha512-fz4spIh+znjO2VjL+IdhEpRJ3YN6sMzITSBijk6FK2UvTqruSQW+/cCZTSNsMiZNvUeq0CqurF+dAbyiGOY6Wg==", + "dependencies": { + "browserify-cipher": "^1.0.0", + "browserify-sign": "^4.0.0", + "create-ecdh": "^4.0.0", + "create-hash": "^1.1.0", + "create-hmac": "^1.1.0", + "diffie-hellman": "^5.0.0", + "inherits": "^2.0.1", + "pbkdf2": "^3.0.3", + "public-encrypt": "^4.0.0", + "randombytes": "^2.0.0", + "randomfill": "^1.0.3" + }, + "engines": { + "node": "*" + } + }, "node_modules/cytoscape": { "version": "3.29.2", "resolved": "https://registry.npmjs.org/cytoscape/-/cytoscape-3.29.2.tgz", @@ -2373,6 +3058,11 @@ "lodash-es": "^4.17.21" } }, + "node_modules/dash-ast": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/dash-ast/-/dash-ast-1.0.0.tgz", + "integrity": "sha512-Vy4dx7gquTeMcQR/hDkYLGUnwVil6vk4FOOct+djUnHOUWt+zJPJAaRIXaAFkPXtJjvlY7o3rfRu0/3hpnwoUA==" + }, "node_modules/dayjs": { "version": "1.11.11", "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.11.11.tgz", @@ -2420,6 +3110,12 @@ } } }, + "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/deepmerge": { "version": "4.3.1", "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.3.1.tgz", @@ -2429,6 +3125,46 @@ "node": ">=0.10.0" } }, + "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==", + "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==", + "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/defined": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/defined/-/defined-1.0.1.tgz", + "integrity": "sha512-hsBd2qSVCRE+5PmNdHt1uzyrFu5d3RwmFDKzyNZMFq/EwDNJF7Ee5+D5oEKF0hU6LhtoUF1macFvOe4AskQC1Q==", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/delaunator": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/delaunator/-/delaunator-5.0.1.tgz", @@ -2437,6 +3173,20 @@ "robust-predicates": "^3.0.2" } }, + "node_modules/deps-sort": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/deps-sort/-/deps-sort-2.0.1.tgz", + "integrity": "sha512-1orqXQr5po+3KI6kQb9A4jnXT1PBwggGl2d7Sq2xsnOeI9GPcE/tGcF9UiSZtZBM7MukY4cAh7MemS6tZYipfw==", + "dependencies": { + "JSONStream": "^1.0.3", + "shasum-object": "^1.0.0", + "subarg": "^1.0.0", + "through2": "^2.0.0" + }, + "bin": { + "deps-sort": "bin/cmd.js" + } + }, "node_modules/dequal": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", @@ -2445,6 +3195,15 @@ "node": ">=6" } }, + "node_modules/des.js": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/des.js/-/des.js-1.1.0.tgz", + "integrity": "sha512-r17GxjhUCjSRy8aiJpr8/UadFIzMzJGexI3Nmz4ADi9LYSFx4gTBp80+NaX/YsXWWLhpZ7v/v/ubEc/bCNfKwg==", + "dependencies": { + "inherits": "^2.0.1", + "minimalistic-assert": "^1.0.0" + } + }, "node_modules/detect-newline": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz", @@ -2454,6 +3213,22 @@ "node": ">=8" } }, + "node_modules/detective": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/detective/-/detective-5.2.1.tgz", + "integrity": "sha512-v9XE1zRnz1wRtgurGu0Bs8uHKFSTdteYZNbIPFVhUZ39L/S79ppMpdmVOZAnoz1jfEFodc48n6MX483Xo3t1yw==", + "dependencies": { + "acorn-node": "^1.8.2", + "defined": "^1.0.0", + "minimist": "^1.2.6" + }, + "bin": { + "detective": "bin/detective.js" + }, + "engines": { + "node": ">=0.8.0" + } + }, "node_modules/diff": { "version": "5.2.0", "resolved": "https://registry.npmjs.org/diff/-/diff-5.2.0.tgz", @@ -2462,11 +3237,64 @@ "node": ">=0.3.1" } }, + "node_modules/diffie-hellman": { + "version": "5.0.3", + "resolved": "https://registry.npmjs.org/diffie-hellman/-/diffie-hellman-5.0.3.tgz", + "integrity": "sha512-kqag/Nl+f3GwyK25fhUMYj81BUOrZ9IuJsjIcDE5icNM9FJHAVm3VcUDxdLPoQtTuUylWm6ZIknYJwwaPxsUzg==", + "dependencies": { + "bn.js": "^4.1.0", + "miller-rabin": "^4.0.0", + "randombytes": "^2.0.0" + } + }, + "node_modules/diffie-hellman/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==" + }, + "node_modules/doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "dependencies": { + "esutils": "^2.0.2" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/doctrine/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/domain-browser": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/domain-browser/-/domain-browser-1.2.0.tgz", + "integrity": "sha512-jnjyiM6eRyZl2H+W8Q/zLMA481hzi0eszAaBUzIVnmYVDBbnLxVNnfu1HgEBvCbL+71FrxMl3E6lpKH7Ge3OXA==", + "engines": { + "node": ">=0.4", + "npm": ">=1.2" + } + }, "node_modules/dompurify": { "version": "3.1.5", "resolved": "https://registry.npmjs.org/dompurify/-/dompurify-3.1.5.tgz", "integrity": "sha512-lwG+n5h8QNpxtyrJW/gJWckL+1/DQiYMX8f7t8Z2AZTPw1esVrqjI63i7Zc2Gz0aKzLVMYC1V1PL/ky+aY/NgA==" }, + "node_modules/duplexer2": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/duplexer2/-/duplexer2-0.1.4.tgz", + "integrity": "sha512-asLFVfWWtJ90ZyOUHMqk7/S2w2guQKxUI2itj3d92ADHhxUSbCMGi1f1cBcJ7xM1To+pE/Khbwo1yuNbMEPKeA==", + "dependencies": { + "readable-stream": "^2.0.2" + } + }, "node_modules/ebnf-parser": { "version": "0.1.10", "resolved": "https://registry.npmjs.org/ebnf-parser/-/ebnf-parser-0.1.10.tgz", @@ -2483,6 +3311,30 @@ "resolved": "https://registry.npmjs.org/elkjs/-/elkjs-0.9.3.tgz", "integrity": "sha512-f/ZeWvW/BCXbhGEf1Ujp29EASo/lk1FDnETgNKwJrsVvGZhUWCZyg3xLJjAsxfOmt8KjswHmI5EwCQcPMpOYhQ==" }, + "node_modules/elliptic": { + "version": "6.5.5", + "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.5.tgz", + "integrity": "sha512-7EjbcmUm17NQFu4Pmgmq2olYMj8nwMnpcddByChSUjArp8F5DQWcIcpriwO4ZToLNAJig0yiyjswfyGNje/ixw==", + "dependencies": { + "bn.js": "^4.11.9", + "brorand": "^1.1.0", + "hash.js": "^1.0.0", + "hmac-drbg": "^1.0.1", + "inherits": "^2.0.4", + "minimalistic-assert": "^1.0.1", + "minimalistic-crypto-utils": "^1.0.1" + } + }, + "node_modules/elliptic/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==" + }, + "node_modules/elliptic/node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, "node_modules/emittery": { "version": "0.13.1", "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.13.1.tgz", @@ -2505,11 +3357,29 @@ "version": "1.3.2", "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", - "dev": true, "dependencies": { "is-arrayish": "^0.2.1" } }, + "node_modules/es-define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.0.tgz", + "integrity": "sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==", + "dependencies": { + "get-intrinsic": "^1.2.4" + }, + "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==", + "engines": { + "node": ">= 0.4" + } + }, "node_modules/escalade": { "version": "3.1.2", "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.2.tgz", @@ -2548,133 +3418,383 @@ "source-map": "~0.1.33" } }, - "node_modules/esprima": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/esprima/-/esprima-1.1.1.tgz", - "integrity": "sha512-qxxB994/7NtERxgXdFgLHIs9M6bhLXc6qtUmWZ3L8+gTQ9qaoyki2887P2IqAYsoENyr8SUbTutStDniOHSDHg==", - "bin": { - "esparse": "bin/esparse.js", - "esvalidate": "bin/esvalidate.js" - }, - "engines": { - "node": ">=0.4.0" - } - }, - "node_modules/estraverse": { - "version": "1.5.1", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-1.5.1.tgz", - "integrity": "sha512-FpCjJDfmo3vsc/1zKSeqR5k42tcIhxFIlvq+h9j0fO2q/h2uLKyweq7rYJ+0CoVvrGQOxIS5wyBrW/+vF58BUQ==", - "engines": { - "node": ">=0.4.0" - } - }, - "node_modules/esutils": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/esutils/-/esutils-1.0.0.tgz", - "integrity": "sha512-x/iYH53X3quDwfHRz4y8rn4XcEwwCJeWsul9pF1zldMbGtgOtMNBEOuYWwB1EQlK2LRa1fev3YAgym/RElp5Cg==", - "engines": { - "node": ">=0.10.0" - } - }, - "node_modules/execa": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/execa/-/execa-5.1.1.tgz", - "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==", + "node_modules/eslint": { + "version": "6.8.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-6.8.0.tgz", + "integrity": "sha512-K+Iayyo2LtyYhDSYwz5D5QdWw0hCacNzyq1Y821Xna2xSJj7cijoLLYmLxTQgcgZ9mC61nryMy9S7GRbYpI5Ig==", "dev": true, "dependencies": { - "cross-spawn": "^7.0.3", - "get-stream": "^6.0.0", - "human-signals": "^2.1.0", - "is-stream": "^2.0.0", - "merge-stream": "^2.0.0", - "npm-run-path": "^4.0.1", - "onetime": "^5.1.2", - "signal-exit": "^3.0.3", - "strip-final-newline": "^2.0.0" + "@babel/code-frame": "^7.0.0", + "ajv": "^6.10.0", + "chalk": "^2.1.0", + "cross-spawn": "^6.0.5", + "debug": "^4.0.1", + "doctrine": "^3.0.0", + "eslint-scope": "^5.0.0", + "eslint-utils": "^1.4.3", + "eslint-visitor-keys": "^1.1.0", + "espree": "^6.1.2", + "esquery": "^1.0.1", + "esutils": "^2.0.2", + "file-entry-cache": "^5.0.1", + "functional-red-black-tree": "^1.0.1", + "glob-parent": "^5.0.0", + "globals": "^12.1.0", + "ignore": "^4.0.6", + "import-fresh": "^3.0.0", + "imurmurhash": "^0.1.4", + "inquirer": "^7.0.0", + "is-glob": "^4.0.0", + "js-yaml": "^3.13.1", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.3.0", + "lodash": "^4.17.14", + "minimatch": "^3.0.4", + "mkdirp": "^0.5.1", + "natural-compare": "^1.4.0", + "optionator": "^0.8.3", + "progress": "^2.0.0", + "regexpp": "^2.0.1", + "semver": "^6.1.2", + "strip-ansi": "^5.2.0", + "strip-json-comments": "^3.0.1", + "table": "^5.2.3", + "text-table": "^0.2.0", + "v8-compile-cache": "^2.0.3" + }, + "bin": { + "eslint": "bin/eslint.js" }, "engines": { - "node": ">=10" + "node": "^8.10.0 || ^10.13.0 || >=11.10.1" }, "funding": { - "url": "https://github.com/sindresorhus/execa?sponsor=1" + "url": "https://opencollective.com/eslint" } }, - "node_modules/execa/node_modules/cross-spawn": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", - "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "node_modules/eslint-scope": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", + "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", "dev": true, "dependencies": { - "path-key": "^3.1.0", - "shebang-command": "^2.0.0", - "which": "^2.0.1" + "esrecurse": "^4.3.0", + "estraverse": "^4.1.1" }, "engines": { - "node": ">= 8" + "node": ">=8.0.0" } }, - "node_modules/execa/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==", + "node_modules/eslint-scope/node_modules/estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", "dev": true, "engines": { - "node": ">=8" + "node": ">=4.0" } }, - "node_modules/execa/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==", + "node_modules/eslint-utils": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-1.4.3.tgz", + "integrity": "sha512-fbBN5W2xdY45KulGXmLHZ3c3FHfVYmKg0IrAKGOkT/464PQsx2UeIzfz1RmEci+KLm1bBaAzZAh8+/E+XAeZ8Q==", "dev": true, "dependencies": { - "shebang-regex": "^3.0.0" + "eslint-visitor-keys": "^1.1.0" }, "engines": { - "node": ">=8" + "node": ">=6" } }, - "node_modules/execa/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==", + "node_modules/eslint-visitor-keys": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", + "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", "dev": true, "engines": { - "node": ">=8" + "node": ">=4" } }, - "node_modules/execa/node_modules/which": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", - "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "node_modules/eslint/node_modules/ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", "dev": true, "dependencies": { - "isexe": "^2.0.0" - }, - "bin": { - "node-which": "bin/node-which" + "color-convert": "^1.9.0" }, "engines": { - "node": ">= 8" + "node": ">=4" } }, - "node_modules/exit": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", - "integrity": "sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==", + "node_modules/eslint/node_modules/chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", "dev": true, + "dependencies": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + }, "engines": { - "node": ">= 0.8.0" + "node": ">=4" } }, - "node_modules/expect": { - "version": "29.7.0", - "resolved": "https://registry.npmjs.org/expect/-/expect-29.7.0.tgz", - "integrity": "sha512-2Zks0hf1VLFYI1kbh0I5jP3KHHyCHpkfyHBzsSXRFgl/Bg9mWYfMW8oD+PdMPlEwy5HNsR9JutYy6pMeOh61nw==", + "node_modules/eslint/node_modules/color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", "dev": true, "dependencies": { - "@jest/expect-utils": "^29.7.0", - "jest-get-type": "^29.6.3", + "color-name": "1.1.3" + } + }, + "node_modules/eslint/node_modules/color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", + "dev": true + }, + "node_modules/eslint/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/eslint/node_modules/has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint/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, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/eslint/node_modules/supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "dependencies": { + "has-flag": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/espree": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-6.2.1.tgz", + "integrity": "sha512-ysCxRQY3WaXJz9tdbWOwuWr5Y/XrPTGX9Kiz3yoUXwW0VZ4w30HTkQLaGx/+ttFjF8i+ACbArnB4ce68a9m5hw==", + "dev": true, + "dependencies": { + "acorn": "^7.1.1", + "acorn-jsx": "^5.2.0", + "eslint-visitor-keys": "^1.1.0" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/esprima": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-1.1.1.tgz", + "integrity": "sha512-qxxB994/7NtERxgXdFgLHIs9M6bhLXc6qtUmWZ3L8+gTQ9qaoyki2887P2IqAYsoENyr8SUbTutStDniOHSDHg==", + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "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/esquery/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/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/esrecurse/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/estraverse": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-1.5.1.tgz", + "integrity": "sha512-FpCjJDfmo3vsc/1zKSeqR5k42tcIhxFIlvq+h9j0fO2q/h2uLKyweq7rYJ+0CoVvrGQOxIS5wyBrW/+vF58BUQ==", + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/esutils": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-1.0.0.tgz", + "integrity": "sha512-x/iYH53X3quDwfHRz4y8rn4XcEwwCJeWsul9pF1zldMbGtgOtMNBEOuYWwB1EQlK2LRa1fev3YAgym/RElp5Cg==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/events": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/events/-/events-3.3.0.tgz", + "integrity": "sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==", + "engines": { + "node": ">=0.8.x" + } + }, + "node_modules/evp_bytestokey": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/evp_bytestokey/-/evp_bytestokey-1.0.3.tgz", + "integrity": "sha512-/f2Go4TognH/KvCISP7OUsHn85hT9nUkxxA9BEWxFn+Oj9o8ZNLm/40hdlgSLyuOimsrTKLUMEorQexp/aPQeA==", + "dependencies": { + "md5.js": "^1.3.4", + "safe-buffer": "^5.1.1" + } + }, + "node_modules/execa": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/execa/-/execa-5.1.1.tgz", + "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==", + "dev": true, + "dependencies": { + "cross-spawn": "^7.0.3", + "get-stream": "^6.0.0", + "human-signals": "^2.1.0", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.1", + "onetime": "^5.1.2", + "signal-exit": "^3.0.3", + "strip-final-newline": "^2.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sindresorhus/execa?sponsor=1" + } + }, + "node_modules/execa/node_modules/cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/execa/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/execa/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/execa/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/execa/node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/exit": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", + "integrity": "sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/expect": { + "version": "29.7.0", + "resolved": "https://registry.npmjs.org/expect/-/expect-29.7.0.tgz", + "integrity": "sha512-2Zks0hf1VLFYI1kbh0I5jP3KHHyCHpkfyHBzsSXRFgl/Bg9mWYfMW8oD+PdMPlEwy5HNsR9JutYy6pMeOh61nw==", + "dev": true, + "dependencies": { + "@jest/expect-utils": "^29.7.0", + "jest-get-type": "^29.6.3", "jest-matcher-utils": "^29.7.0", "jest-message-util": "^29.7.0", "jest-util": "^29.7.0" @@ -2692,12 +3812,55 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, + "node_modules/external-editor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-3.1.0.tgz", + "integrity": "sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==", + "dev": true, + "dependencies": { + "chardet": "^0.7.0", + "iconv-lite": "^0.4.24", + "tmp": "^0.0.33" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/external-editor/node_modules/iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dev": true, + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "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/fast-safe-stringify": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/fast-safe-stringify/-/fast-safe-stringify-2.1.1.tgz", + "integrity": "sha512-W+KJc2dmILlPplD/H4K9l9LcAHAfPtP6BY84uVLXQ6Evcz9Lcg33Y2z1IVblT6xdY54PXYVHEv+0Wpq8Io6zkA==" + }, "node_modules/fb-watchman": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/fb-watchman/-/fb-watchman-2.0.2.tgz", @@ -2718,6 +3881,33 @@ "node": ">= 0.4.0" } }, + "node_modules/figures": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-3.2.0.tgz", + "integrity": "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==", + "dev": true, + "dependencies": { + "escape-string-regexp": "^1.0.5" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/file-entry-cache": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-5.0.1.tgz", + "integrity": "sha512-bCg29ictuBaKUwwArK4ouCaqDgLZcysCFLmM/Yn/FDoqndh/9vNuQfXRDvTuXKLxfD/JtZQGKFT8MGcJBK644g==", + "dev": true, + "dependencies": { + "flat-cache": "^2.0.1" + }, + "engines": { + "node": ">=4" + } + }, "node_modules/fill-range": { "version": "7.1.1", "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", @@ -2743,11 +3933,38 @@ "node": ">=8" } }, + "node_modules/flat-cache": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-2.0.1.tgz", + "integrity": "sha512-LoQe6yDuUMDzQAEH8sgmh4Md6oZnc/7PjtwjNFSzveXqSHt6ka9fPBuso7IGf9Rz4uqnSnWiFH2B/zj24a5ReA==", + "dev": true, + "dependencies": { + "flatted": "^2.0.0", + "rimraf": "2.6.3", + "write": "1.0.3" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/flatted": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-2.0.2.tgz", + "integrity": "sha512-r5wGx7YeOwNWNlCA0wQ86zKyDLMQr+/RB8xy74M4hTphfmjlijTSSXGuH8rnvKZnfT9i+75zmd8jcKdMR4O6jA==", + "dev": true + }, + "node_modules/for-each": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/for-each/-/for-each-0.3.3.tgz", + "integrity": "sha512-jqYfLp7mo9vIyQf8ykW2v7A+2N4QjeCeI5+Dz9XraiO1ign81wjiH7Fb9vSOWvQfNtmSa4H2RoQTrrXivdUZmw==", + "dependencies": { + "is-callable": "^1.1.3" + } + }, "node_modules/fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", - "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", - "dev": true + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==" }, "node_modules/fsevents": { "version": "2.3.3", @@ -2767,11 +3984,16 @@ "version": "1.1.2", "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", - "dev": true, "funding": { "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/functional-red-black-tree": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", + "integrity": "sha512-dsKNQNdj6xA3T+QlADDA7mOSlX0qiMINjn0cgr+eGHGsbSHzTabcIogz2+p/iqP1Xs6EP/sS2SbqH+brGTbq0g==", + "dev": true + }, "node_modules/gensync": { "version": "1.0.0-beta.2", "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", @@ -2781,6 +4003,11 @@ "node": ">=6.9.0" } }, + "node_modules/get-assigned-identifiers": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/get-assigned-identifiers/-/get-assigned-identifiers-1.2.0.tgz", + "integrity": "sha512-mBBwmeGTrxEMO4pMaaf/uUEFHnYtwr8FTe8Y/mer4rcV/bye0qGm6pw1bGZFGStxC5O76c5ZAVBGnqHmOaJpdQ==" + }, "node_modules/get-caller-file": { "version": "2.0.5", "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", @@ -2790,6 +4017,24 @@ "node": "6.* || 8.* || >= 10.*" } }, + "node_modules/get-intrinsic": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.4.tgz", + "integrity": "sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==", + "dependencies": { + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "has-proto": "^1.0.1", + "has-symbols": "^1.0.3", + "hasown": "^2.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/get-package-type": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/get-package-type/-/get-package-type-0.1.0.tgz", @@ -2816,7 +4061,6 @@ "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz", "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", "deprecated": "Glob versions prior to v9 are no longer supported", - "dev": true, "dependencies": { "fs.realpath": "^1.0.0", "inflight": "^1.0.4", @@ -2832,12 +4076,58 @@ "url": "https://github.com/sponsors/isaacs" } }, + "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, + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/globals": { + "version": "12.4.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz", + "integrity": "sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg==", + "dev": true, + "dependencies": { + "type-fest": "^0.8.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/gopd": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.0.1.tgz", + "integrity": "sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==", + "dependencies": { + "get-intrinsic": "^1.1.3" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/graceful-fs": { "version": "4.2.11", "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", "dev": true }, + "node_modules/has": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/has/-/has-1.0.4.tgz", + "integrity": "sha512-qdSAmqLF6209RFj4VVItywPMbm3vWylknmB3nvNiUIs72xAimcM8nVYxYr7ncvZq5qzk9MKIZR8ijqD/1QuYjQ==", + "engines": { + "node": ">= 0.4.0" + } + }, "node_modules/has-flag": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", @@ -2847,11 +4137,78 @@ "node": ">=8" } }, + "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==", + "dependencies": { + "es-define-property": "^1.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-proto": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.3.tgz", + "integrity": "sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-symbols": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", + "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", + "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==", + "dependencies": { + "has-symbols": "^1.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/hash-base": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/hash-base/-/hash-base-3.0.4.tgz", + "integrity": "sha512-EeeoJKjTyt868liAlVmcv2ZsUfGHlE3Q+BICOXcZiwN3osr5Q/zFGYmTJpoIzuaSTAwndFy+GqhEwlU4L3j4Ow==", + "dependencies": { + "inherits": "^2.0.1", + "safe-buffer": "^5.0.1" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/hash.js": { + "version": "1.1.7", + "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.7.tgz", + "integrity": "sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==", + "dependencies": { + "inherits": "^2.0.3", + "minimalistic-assert": "^1.0.1" + } + }, "node_modules/hasown": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", - "dev": true, "dependencies": { "function-bind": "^1.1.2" }, @@ -2859,12 +4216,35 @@ "node": ">= 0.4" } }, + "node_modules/hmac-drbg": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz", + "integrity": "sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==", + "dependencies": { + "hash.js": "^1.0.3", + "minimalistic-assert": "^1.0.0", + "minimalistic-crypto-utils": "^1.0.1" + } + }, "node_modules/html-escaper": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", "dev": true }, + "node_modules/htmlescape": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/htmlescape/-/htmlescape-1.1.1.tgz", + "integrity": "sha512-eVcrzgbR4tim7c7soKQKtxa/kQM4TzjnlU83rcZ9bHU6t31ehfV7SktN6McWgwPWg+JYMA/O3qpGxBvFq1z2Jg==", + "engines": { + "node": ">=0.10" + } + }, + "node_modules/https-browserify": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/https-browserify/-/https-browserify-1.0.0.tgz", + "integrity": "sha512-J+FkSdyD+0mA0N+81tMotaRMfSL9SGi+xpD3T6YApKsc3bGSXJlfXri3VyFOeYkfLRQisDk1W+jIFFKBeUBbBg==" + }, "node_modules/human-signals": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz", @@ -2885,6 +4265,50 @@ "node": ">=0.10.0" } }, + "node_modules/ieee754": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", + "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/ignore": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", + "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/import-fresh": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", + "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", + "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/import-local": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.1.0.tgz", @@ -2918,7 +4342,6 @@ "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", "deprecated": "This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.", - "dev": true, "dependencies": { "once": "^1.3.0", "wrappy": "1" @@ -2929,6 +4352,94 @@ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", "integrity": "sha512-x00IRNXNy63jwGkJmzPigoySHbaqpNuzKbBOmzK+g2OdZpQ9w+sxCN+VSB3ja7IAge2OP2qpfxTjeNcyjmW1uw==" }, + "node_modules/inline-source-map": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/inline-source-map/-/inline-source-map-0.6.3.tgz", + "integrity": "sha512-1aVsPEsJWMJq/pdMU61CDlm1URcW702MTB4w9/zUjMus6H/Py8o7g68Pr9D4I6QluWGt/KdmswuRhaA05xVR1w==", + "dependencies": { + "source-map": "~0.5.3" + } + }, + "node_modules/inline-source-map/node_modules/source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha512-LbrmJOMUSdEVxIKvdcJzQC+nQhe8FUZQTXQy6+I75skNgn3OoQ0DZA8YnFa7gp8tqtL3KPf1kmo0R5DoApeSGQ==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/inquirer": { + "version": "7.3.3", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-7.3.3.tgz", + "integrity": "sha512-JG3eIAj5V9CwcGvuOmoo6LB9kbAYT8HXffUl6memuszlwDC/qvFAJw49XJ5NROSFNPxp3iQg1GqkFhaY/CR0IA==", + "dev": true, + "dependencies": { + "ansi-escapes": "^4.2.1", + "chalk": "^4.1.0", + "cli-cursor": "^3.1.0", + "cli-width": "^3.0.0", + "external-editor": "^3.0.3", + "figures": "^3.0.0", + "lodash": "^4.17.19", + "mute-stream": "0.0.8", + "run-async": "^2.4.0", + "rxjs": "^6.6.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0", + "through": "^2.3.6" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/inquirer/node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/inquirer/node_modules/strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/insert-module-globals": { + "version": "7.2.1", + "resolved": "https://registry.npmjs.org/insert-module-globals/-/insert-module-globals-7.2.1.tgz", + "integrity": "sha512-ufS5Qq9RZN+Bu899eA9QCAYThY+gGW7oRkmb0vC93Vlyu/CFGcH0OYPEjVkDXA5FEbTt1+VWzdoOD3Ny9N+8tg==", + "dependencies": { + "acorn-node": "^1.5.2", + "combine-source-map": "^0.8.0", + "concat-stream": "^1.6.1", + "is-buffer": "^1.1.0", + "JSONStream": "^1.0.3", + "path-is-absolute": "^1.0.1", + "process": "~0.11.0", + "through2": "^2.0.0", + "undeclared-identifiers": "^1.1.2", + "xtend": "^4.0.0" + }, + "bin": { + "insert-module-globals": "bin/cmd.js" + } + }, "node_modules/internmap": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/internmap/-/internmap-2.0.3.tgz", @@ -2937,17 +4448,46 @@ "node": ">=12" } }, + "node_modules/is-arguments": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.1.1.tgz", + "integrity": "sha512-8Q7EARjzEnKpt/PCD7e1cgUS0a6X8u5tdSiMqXhojOdoV9TsMsiO+9VLC5vAmO8N7/GmXn7yjR8qnA6bVAEzfA==", + "dependencies": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/is-arrayish": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", - "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", - "dev": true + "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==" + }, + "node_modules/is-buffer": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", + "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==" + }, + "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==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } }, "node_modules/is-core-module": { "version": "2.14.0", "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.14.0.tgz", "integrity": "sha512-a5dFJih5ZLYlRtDc0dZWP7RiKr6xIKzmn/oAYCDvdLThadVgyJwlaoQPmRtMSpz+rk0OGAgIu+TcM9HUF0fk1A==", - "dev": true, "dependencies": { "hasown": "^2.0.2" }, @@ -2958,6 +4498,15 @@ "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-fullwidth-code-point": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", @@ -2976,6 +4525,32 @@ "node": ">=6" } }, + "node_modules/is-generator-function": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/is-generator-function/-/is-generator-function-1.0.10.tgz", + "integrity": "sha512-jsEjy9l3yiXEQ+PsXdmBwEPcOxaXWLspKdplFUVI9vq1iZgIekeC0L167qeu86czQaxed3q/Uzuw0swL0irL8A==", + "dependencies": { + "has-tostringtag": "^1.0.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-number": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", @@ -2997,6 +4572,30 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/is-typed-array": { + "version": "1.1.13", + "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.13.tgz", + "integrity": "sha512-uZ25/bUAlUY5fR4OKT4rZQEBrzQWYV9ZJYGGsUmEJ6thodVJ1HX64ePQ6Z0qPWP+m+Uq6e9UugrE38jeYsDSMw==", + "dependencies": { + "which-typed-array": "^1.1.14" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-utf8": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-utf8/-/is-utf8-0.2.1.tgz", + "integrity": "sha512-rMYPYvCzsXywIsldgLaSoPlw5PfoB/ssr7hY4pLfcodrA5M/eArza1a9VmTiNIBNMjOGr1Ow9mTyU2o69U6U9Q==" + }, + "node_modules/isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==" + }, "node_modules/isexe": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", @@ -4062,6 +5661,18 @@ "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", "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": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true + }, "node_modules/json5": { "version": "2.2.3", "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", @@ -4089,6 +5700,14 @@ "node": ">= 0.6" } }, + "node_modules/jsonparse": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/jsonparse/-/jsonparse-1.3.1.tgz", + "integrity": "sha512-POQXvpdL69+CluYsillJ7SUhKvytYjW9vG/GKpnf+xP8UWgYEM/RaMzHHofbALDiKbbP1W8UEYmgGl39WkPZsg==", + "engines": [ + "node >= 0.2.0" + ] + }, "node_modules/JSONSelect": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/JSONSelect/-/JSONSelect-0.4.0.tgz", @@ -4097,6 +5716,21 @@ "node": ">=0.4.7" } }, + "node_modules/JSONStream": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/JSONStream/-/JSONStream-1.3.5.tgz", + "integrity": "sha512-E+iruNOY8VV9s4JEbe1aNEm6MiszPRr/UfcHMz0TQh1BXSxHK+ASV1R6W4HpjBhSeS+54PIsAMCBmwD06LLsqQ==", + "dependencies": { + "jsonparse": "^1.2.0", + "through": ">=2.2.7 <3" + }, + "bin": { + "JSONStream": "bin.js" + }, + "engines": { + "node": "*" + } + }, "node_modules/JSV": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/JSV/-/JSV-4.0.2.tgz", @@ -4141,6 +5775,15 @@ "node": ">=6" } }, + "node_modules/labeled-stream-splicer": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/labeled-stream-splicer/-/labeled-stream-splicer-2.0.2.tgz", + "integrity": "sha512-Ca4LSXFFZUjPScRaqOcFxneA0VpKZr4MMYCljyQr4LIewTLb3Y0IUTIsnBBsVubIeEfxeSZpSjSsRM8APEQaAw==", + "dependencies": { + "inherits": "^2.0.1", + "stream-splicer": "^2.0.0" + } + }, "node_modules/layout-base": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/layout-base/-/layout-base-1.0.2.tgz", @@ -4155,6 +5798,19 @@ "node": ">=6" } }, + "node_modules/levn": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", + "integrity": "sha512-0OO4y2iOHix2W6ujICbKIaEQXvFQHue65vUG3pb5EUomzPI90z9hsA1VsO/dbIIpC53J8gxM9Q4Oho0jrCM/yA==", + "dev": true, + "dependencies": { + "prelude-ls": "~1.1.2", + "type-check": "~0.3.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, "node_modules/lex-parser": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/lex-parser/-/lex-parser-0.1.4.tgz", @@ -4188,6 +5844,11 @@ "resolved": "https://registry.npmjs.org/lodash-es/-/lodash-es-4.17.21.tgz", "integrity": "sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw==" }, + "node_modules/lodash.memoize": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-3.0.4.tgz", + "integrity": "sha512-eDn9kqrAmVUC1wmZvlQ6Uhde44n+tXpqPrN8olQJbttgh0oKclk+SF54P47VEGE9CEiMeRwAP8BaM7UHvBkz2A==" + }, "node_modules/lru-cache": { "version": "5.1.1", "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", @@ -4227,6 +5888,16 @@ "tmpl": "1.0.5" } }, + "node_modules/md5.js": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.5.tgz", + "integrity": "sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg==", + "dependencies": { + "hash-base": "^3.0.0", + "inherits": "^2.0.1", + "safe-buffer": "^5.1.2" + } + }, "node_modules/mdast-util-from-markdown": { "version": "1.3.1", "resolved": "https://registry.npmjs.org/mdast-util-from-markdown/-/mdast-util-from-markdown-1.3.1.tgz", @@ -4729,6 +6400,23 @@ "node": ">=8.6" } }, + "node_modules/miller-rabin": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/miller-rabin/-/miller-rabin-4.0.1.tgz", + "integrity": "sha512-115fLhvZVqWwHPbClyntxEVfVDfl9DLLTuJvq3g2O/Oxi8AiNouAHvDSzHS0viUJc+V5vm3eq91Xwqn9dp4jRA==", + "dependencies": { + "bn.js": "^4.0.0", + "brorand": "^1.0.1" + }, + "bin": { + "miller-rabin": "bin/miller-rabin" + } + }, + "node_modules/miller-rabin/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==" + }, "node_modules/mimic-fn": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", @@ -4738,11 +6426,20 @@ "node": ">=6" } }, + "node_modules/minimalistic-assert": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz", + "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==" + }, + "node_modules/minimalistic-crypto-utils": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz", + "integrity": "sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg==" + }, "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" }, @@ -4750,6 +6447,59 @@ "node": "*" } }, + "node_modules/minimist": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", + "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/mkdirp": { + "version": "0.5.6", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.6.tgz", + "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==", + "dev": true, + "dependencies": { + "minimist": "^1.2.6" + }, + "bin": { + "mkdirp": "bin/cmd.js" + } + }, + "node_modules/mkdirp-classic": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz", + "integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==" + }, + "node_modules/module-deps": { + "version": "6.2.3", + "resolved": "https://registry.npmjs.org/module-deps/-/module-deps-6.2.3.tgz", + "integrity": "sha512-fg7OZaQBcL4/L+AK5f4iVqf9OMbCclXfy/znXRxTVhJSeW5AIlS9AwheYwDaXM3lVW7OBeaeUEY3gbaC6cLlSA==", + "dependencies": { + "browser-resolve": "^2.0.0", + "cached-path-relative": "^1.0.2", + "concat-stream": "~1.6.0", + "defined": "^1.0.0", + "detective": "^5.2.0", + "duplexer2": "^0.1.2", + "inherits": "^2.0.1", + "JSONStream": "^1.0.3", + "parents": "^1.0.0", + "readable-stream": "^2.0.2", + "resolve": "^1.4.0", + "stream-combiner2": "^1.1.1", + "subarg": "^1.0.0", + "through2": "^2.0.0", + "xtend": "^4.0.0" + }, + "bin": { + "module-deps": "bin/cmd.js" + }, + "engines": { + "node": ">= 0.8.0" + } + }, "node_modules/moment-mini": { "version": "2.29.4", "resolved": "https://registry.npmjs.org/moment-mini/-/moment-mini-2.29.4.tgz", @@ -4768,12 +6518,24 @@ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" }, + "node_modules/mute-stream": { + "version": "0.0.8", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.8.tgz", + "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", + "dev": true + }, "node_modules/natural-compare": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", "dev": true }, + "node_modules/nice-try": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", + "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", + "dev": true + }, "node_modules/node-int64": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", @@ -4834,11 +6596,54 @@ "node": ">=8" } }, + "node_modules/object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-inspect": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.2.tgz", + "integrity": "sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==", + "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==", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/object.assign": { + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.5.tgz", + "integrity": "sha512-byy+U7gp+FVwmyzKPYhW2h5l3crpmGsxl7X2s8y43IgxvG4g3QZ6CffDtsNQy1WsmZpQbO+ybo0AlW7TY6DcBQ==", + "dependencies": { + "call-bind": "^1.0.5", + "define-properties": "^1.2.1", + "has-symbols": "^1.0.3", + "object-keys": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/once": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", - "dev": true, "dependencies": { "wrappy": "1" } @@ -4849,13 +6654,44 @@ "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", "dev": true, "dependencies": { - "mimic-fn": "^2.1.0" + "mimic-fn": "^2.1.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/optionator": { + "version": "0.8.3", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz", + "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==", + "dev": true, + "dependencies": { + "deep-is": "~0.1.3", + "fast-levenshtein": "~2.0.6", + "levn": "~0.3.0", + "prelude-ls": "~1.1.2", + "type-check": "~0.3.2", + "word-wrap": "~1.2.3" }, "engines": { - "node": ">=6" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" + "node": ">= 0.8.0" + } + }, + "node_modules/os-browserify": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/os-browserify/-/os-browserify-0.3.0.tgz", + "integrity": "sha512-gjcpUc3clBf9+210TRaDWbf+rZZZEshZ+DlXMRCeAjp0xhTrnQsKHypIy1J3d5hKdUzj69t708EHtU8P6bUn0A==" + }, + "node_modules/os-tmpdir": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", + "integrity": "sha512-D2FR03Vir7FIu45XBY20mTb+/ZSWB00sjU9jdQXt83gDrI4Ztz5Fs7/yy74g2N5SVQY4xY1qDr4rNddwYRVX0g==", + "dev": true, + "engines": { + "node": ">=0.10.0" } }, "node_modules/p-limit": { @@ -4909,6 +6745,47 @@ "node": ">=6" } }, + "node_modules/pako": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.11.tgz", + "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==" + }, + "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/parents": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parents/-/parents-1.0.1.tgz", + "integrity": "sha512-mXKF3xkoUt5td2DoxpLmtOmZvko9VfFpwRwkKDHSNvgmpLAeBo18YDhcPbBzJq+QLCHMbGOfzia2cX4U+0v9Mg==", + "dependencies": { + "path-platform": "~0.11.15" + } + }, + "node_modules/parse-asn1": { + "version": "5.1.7", + "resolved": "https://registry.npmjs.org/parse-asn1/-/parse-asn1-5.1.7.tgz", + "integrity": "sha512-CTM5kuWR3sx9IFamcl5ErfPl6ea/N8IYwiJ+vpeB2g+1iknv7zBl5uPwbMbRVznRVbrNY6lGuDoE5b30grmbqg==", + "dependencies": { + "asn1.js": "^4.10.1", + "browserify-aes": "^1.2.0", + "evp_bytestokey": "^1.0.3", + "hash-base": "~3.0", + "pbkdf2": "^3.1.2", + "safe-buffer": "^5.2.1" + }, + "engines": { + "node": ">= 0.10" + } + }, "node_modules/parse-json": { "version": "5.2.0", "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", @@ -4936,6 +6813,11 @@ "util": "^0.10.3" } }, + "node_modules/path-browserify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-browserify/-/path-browserify-1.0.1.tgz", + "integrity": "sha512-b7uo2UCUOYZcnF/3ID0lulOJi/bafxa1xPe7ZPsammBSpjSWQkjNxlt635YGS2MiR9GjvuXCtz2emr3jbsz98g==" + }, "node_modules/path-exists": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", @@ -4949,16 +6831,46 @@ "version": "1.0.1", "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", - "dev": true, "engines": { "node": ">=0.10.0" } }, + "node_modules/path-key": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", + "integrity": "sha512-fEHGKCSmUSDPv4uoj8AlD+joPlq3peND+HRYyxFz4KPw4z926S/b8rIuFs2FYJg3BwsxJf6A9/3eIdLaYC+9Dw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, "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 + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==" + }, + "node_modules/path-platform": { + "version": "0.11.15", + "resolved": "https://registry.npmjs.org/path-platform/-/path-platform-0.11.15.tgz", + "integrity": "sha512-Y30dB6rab1A/nfEKsZxmr01nUotHX0c/ZiIAsCTatEe1CmS5Pm5He7fZ195bPT7RdquoaL8lLxFCMQi/bS7IJg==", + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/pbkdf2": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.1.2.tgz", + "integrity": "sha512-iuh7L6jA7JEGu2WxDwtQP1ddOpaJNC4KlDEFfdQajSGgGPNi4OyDc2R7QnbY2bR9QjBVGwgvTdNJZoE7RaxUMA==", + "dependencies": { + "create-hash": "^1.1.2", + "create-hmac": "^1.1.4", + "ripemd160": "^2.0.1", + "safe-buffer": "^5.0.1", + "sha.js": "^2.4.8" + }, + "engines": { + "node": ">=0.12" + } }, "node_modules/picocolors": { "version": "1.0.1", @@ -4999,6 +6911,23 @@ "node": ">=8" } }, + "node_modules/possible-typed-array-names": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.0.0.tgz", + "integrity": "sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/prelude-ls": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", + "integrity": "sha512-ESF23V4SKG6lVSGZgYNpbsiaAkdab6ZgOxe52p7+Kid3W3u3bxR4Vfd/o21dmN7jSt0IwgZ4v5MUd26FEtXE9w==", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, "node_modules/pretty-format": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz", @@ -5033,6 +6962,20 @@ "node": ">= 0.6.0" } }, + "node_modules/process-nextick-args": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", + "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==" + }, + "node_modules/progress": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/progress/-/progress-2.0.3.tgz", + "integrity": "sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA==", + "dev": true, + "engines": { + "node": ">=0.4.0" + } + }, "node_modules/prompts": { "version": "2.4.2", "resolved": "https://registry.npmjs.org/prompts/-/prompts-2.4.2.tgz", @@ -5055,6 +6998,29 @@ "node": ">=6" } }, + "node_modules/public-encrypt": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/public-encrypt/-/public-encrypt-4.0.3.tgz", + "integrity": "sha512-zVpa8oKZSz5bTMTFClc1fQOnyyEzpl5ozpi1B5YcvBrdohMjH2rfsBtyXcuNuwjsDIXmBYlF2N5FlJYhR29t8Q==", + "dependencies": { + "bn.js": "^4.1.0", + "browserify-rsa": "^4.0.0", + "create-hash": "^1.1.0", + "parse-asn1": "^5.0.0", + "randombytes": "^2.0.1", + "safe-buffer": "^5.1.2" + } + }, + "node_modules/public-encrypt/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==" + }, + "node_modules/punycode": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", + "integrity": "sha512-jmYNElW7yvO7TV33CjSmvSiE2yco3bV2czu/OzDKdMNVZQWfxCblURLhf+47syQRBntjfLdd/H0egrzIG+oaFQ==" + }, "node_modules/pure-rand": { "version": "6.1.0", "resolved": "https://registry.npmjs.org/pure-rand/-/pure-rand-6.1.0.tgz", @@ -5071,12 +7037,95 @@ } ] }, + "node_modules/qs": { + "version": "6.12.1", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.12.1.tgz", + "integrity": "sha512-zWmv4RSuB9r2mYQw3zxQuHWeU+42aKi1wWig/j4ele4ygELZ7PEO6MM7rim9oAQH2A5MWfsAVf/jPvTPgCbvUQ==", + "dependencies": { + "side-channel": "^1.0.6" + }, + "engines": { + "node": ">=0.6" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/querystring-es3": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/querystring-es3/-/querystring-es3-0.2.1.tgz", + "integrity": "sha512-773xhDQnZBMFobEiztv8LIl70ch5MSF/jUQVlhwFyBILqq96anmoctVIYz+ZRp0qbCKATTn6ev02M3r7Ga5vqA==", + "engines": { + "node": ">=0.4.x" + } + }, + "node_modules/randombytes": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", + "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", + "dependencies": { + "safe-buffer": "^5.1.0" + } + }, + "node_modules/randomfill": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/randomfill/-/randomfill-1.0.4.tgz", + "integrity": "sha512-87lcbR8+MhcWcUiQ+9e+Rwx8MyR2P7qnt15ynUlbm3TU/fjbgz4GsvfSUDTemtCCtVCqb4ZcEFlyPNTh9bBTLw==", + "dependencies": { + "randombytes": "^2.0.5", + "safe-buffer": "^5.1.0" + } + }, "node_modules/react-is": { "version": "18.3.1", "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==", "dev": true }, + "node_modules/read-only-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/read-only-stream/-/read-only-stream-2.0.0.tgz", + "integrity": "sha512-3ALe0bjBVZtkdWKIcThYpQCLbBMd/+Tbh2CDSrAIDO3UsZ4Xs+tnyjv2MjCOMMgBG+AsUOeuP1cgtY1INISc8w==", + "dependencies": { + "readable-stream": "^2.0.2" + } + }, + "node_modules/readable-stream": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", + "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" + } + }, + "node_modules/readable-stream/node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "node_modules/readable-stream/node_modules/string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "dependencies": { + "safe-buffer": "~5.1.0" + } + }, + "node_modules/regexpp": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-2.0.1.tgz", + "integrity": "sha512-lv0M6+TkDVniA3aD1Eg0DVpfU/booSu7Eev3TDO/mZKHBfVjgCGTV4t4buppESEYDtkArYFOxTJWv6S5C+iaNw==", + "dev": true, + "engines": { + "node": ">=6.5.0" + } + }, "node_modules/require-directory": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", @@ -5090,7 +7139,6 @@ "version": "1.22.8", "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.8.tgz", "integrity": "sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==", - "dev": true, "dependencies": { "is-core-module": "^2.13.0", "path-parse": "^1.0.7", @@ -5124,6 +7172,15 @@ "node": ">=8" } }, + "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/resolve.exports": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/resolve.exports/-/resolve.exports-2.0.2.tgz", @@ -5133,16 +7190,72 @@ "node": ">=10" } }, + "node_modules/restore-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-3.1.0.tgz", + "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", + "dev": true, + "dependencies": { + "onetime": "^5.1.0", + "signal-exit": "^3.0.2" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/rimraf": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", + "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", + "deprecated": "Rimraf versions prior to v4 are no longer supported", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/ripemd160": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/ripemd160/-/ripemd160-2.0.2.tgz", + "integrity": "sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==", + "dependencies": { + "hash-base": "^3.0.0", + "inherits": "^2.0.1" + } + }, "node_modules/robust-predicates": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/robust-predicates/-/robust-predicates-3.0.2.tgz", "integrity": "sha512-IXgzBWvWQwE6PrDI05OvmXUIruQTcoMDzRsOd5CDvHCVLcLHMTSYvOK5Cm46kWqlV3yAbuSpBZdJ5oP5OUoStg==" }, + "node_modules/run-async": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.4.1.tgz", + "integrity": "sha512-tvVnVv01b8c1RrA6Ep7JkStj85Guv/YrMcwqYQnwjsAS2cTmmPGBBjAjpCW7RrSodNSoE2/qg9O4bceNvUuDgQ==", + "dev": true, + "engines": { + "node": ">=0.12.0" + } + }, "node_modules/rw": { "version": "1.3.3", "resolved": "https://registry.npmjs.org/rw/-/rw-1.3.3.tgz", "integrity": "sha512-PdhdWy89SiZogBLaw42zdeqtRJ//zFd2PgQavcICDUgJT5oW10QCRKbJ6bg4r0/UY2M6BWd5tkxuGFRvCkgfHQ==" }, + "node_modules/rxjs": { + "version": "6.6.7", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.6.7.tgz", + "integrity": "sha512-hTdwr+7yYNIT5n4AMYp85KA6yw2Va0FLa3Rguvbpa4W3I5xynaBZo41cM3XM+4Q6fRMj3sBYIR1VAmZMXYJvRQ==", + "dev": true, + "dependencies": { + "tslib": "^1.9.0" + }, + "engines": { + "npm": ">=2.0.0" + } + }, "node_modules/sade": { "version": "1.8.1", "resolved": "https://registry.npmjs.org/sade/-/sade-1.8.1.tgz", @@ -5154,6 +7267,25 @@ "node": ">=6" } }, + "node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, "node_modules/safer-buffer": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", @@ -5171,25 +7303,176 @@ "node": ">=10" } }, - "node_modules/signal-exit": { - "version": "3.0.7", - "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", - "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", - "dev": true + "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==", + "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/sha.js": { + "version": "2.4.11", + "resolved": "https://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz", + "integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==", + "dependencies": { + "inherits": "^2.0.1", + "safe-buffer": "^5.0.1" + }, + "bin": { + "sha.js": "bin.js" + } + }, + "node_modules/shasum-object": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/shasum-object/-/shasum-object-1.0.0.tgz", + "integrity": "sha512-Iqo5rp/3xVi6M4YheapzZhhGPVs0yZwHj7wvwQ1B9z8H6zk+FEnI7y3Teq7qwnekfEhu8WmG2z0z4iWZaxLWVg==", + "dependencies": { + "fast-safe-stringify": "^2.0.7" + } + }, + "node_modules/shebang-command": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", + "integrity": "sha512-EV3L1+UQWGor21OmnvojK36mhg+TyIKDh3iFBKBohr5xeXIhNBcx8oWdgkTEEQ+BEFFYdLRuqMfd5L84N1V5Vg==", + "dev": true, + "dependencies": { + "shebang-regex": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/shebang-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", + "integrity": "sha512-wpoSFAxys6b2a2wHZ1XpDSgD7N9iVjg29Ph9uV/uaP9Ex/KXlkTZTeddxDPSYQpgvzKLGJke2UU0AzoGCjNIvQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/shell-quote": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/shell-quote/-/shell-quote-1.8.1.tgz", + "integrity": "sha512-6j1W9l1iAs/4xYBI1SYOVZyFcCis9b4KCLQ8fgAGG07QvzaRLVVRQvAy85yNmmZSjYjg4MWh4gNvlPujU/5LpA==", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.6.tgz", + "integrity": "sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==", + "dependencies": { + "call-bind": "^1.0.7", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.4", + "object-inspect": "^1.13.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/signal-exit": { + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", + "dev": true + }, + "node_modules/simple-concat": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/simple-concat/-/simple-concat-1.0.1.tgz", + "integrity": "sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/sisteransi": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", + "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==", + "dev": true + }, + "node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/slice-ansi": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-2.1.0.tgz", + "integrity": "sha512-Qu+VC3EwYLldKa1fCxuuvULvSJOKEgk9pi8dZeCVK7TqBfUNTH4sFkk4joj8afVSfAYgJoSOetjx9QWOJ5mYoQ==", + "dev": true, + "dependencies": { + "ansi-styles": "^3.2.0", + "astral-regex": "^1.0.0", + "is-fullwidth-code-point": "^2.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/slice-ansi/node_modules/ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "dependencies": { + "color-convert": "^1.9.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/slice-ansi/node_modules/color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dev": true, + "dependencies": { + "color-name": "1.1.3" + } }, - "node_modules/sisteransi": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", - "integrity": "sha512-bLGGlR1QxBcynn2d5YmDX4MGjlZvy2MRBDRNHLJ8VI6l6+9FUiyTFNJ0IveOSP0bcXgVDPRcfGqA0pjaqUpfVg==", + "node_modules/slice-ansi/node_modules/color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", "dev": true }, - "node_modules/slash": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", - "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "node_modules/slice-ansi/node_modules/is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha512-VHskAKYM8RfSFXwee5t5cbN5PZeq1Wrh6qd5bkyiXIf6UQcN6w/A0eXM9r6t8d+GYOh+o6ZhiEnb88LN/Y8m2w==", "dev": true, "engines": { - "node": ">=8" + "node": ">=4" } }, "node_modules/source-map": { @@ -5250,6 +7533,88 @@ "node": ">=8" } }, + "node_modules/stream-browserify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/stream-browserify/-/stream-browserify-3.0.0.tgz", + "integrity": "sha512-H73RAHsVBapbim0tU2JwwOiXUj+fikfiaoYAKHF3VJfA0pe2BCzkhAHBlLG6REzE+2WNZcxOXjK7lkso+9euLA==", + "dependencies": { + "inherits": "~2.0.4", + "readable-stream": "^3.5.0" + } + }, + "node_modules/stream-browserify/node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, + "node_modules/stream-browserify/node_modules/readable-stream": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/stream-combiner2": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/stream-combiner2/-/stream-combiner2-1.1.1.tgz", + "integrity": "sha512-3PnJbYgS56AeWgtKF5jtJRT6uFJe56Z0Hc5Ngg/6sI6rIt8iiMBTa9cvdyFfpMQjaVHr8dusbNeFGIIonxOvKw==", + "dependencies": { + "duplexer2": "~0.1.0", + "readable-stream": "^2.0.2" + } + }, + "node_modules/stream-http": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/stream-http/-/stream-http-3.2.0.tgz", + "integrity": "sha512-Oq1bLqisTyK3TSCXpPbT4sdeYNdmyZJv1LxpEm2vu1ZhK89kSE5YXwZc3cWk0MagGaKriBh9mCFbVGtO+vY29A==", + "dependencies": { + "builtin-status-codes": "^3.0.0", + "inherits": "^2.0.4", + "readable-stream": "^3.6.0", + "xtend": "^4.0.2" + } + }, + "node_modules/stream-http/node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, + "node_modules/stream-http/node_modules/readable-stream": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/stream-splicer": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/stream-splicer/-/stream-splicer-2.0.1.tgz", + "integrity": "sha512-Xizh4/NPuYSyAXyT7g8IvdJ9HJpxIGL9PjyhtywCZvvP0OPIdqyrr4dMikeuvY8xahpdKEBlBTySe583totajg==", + "dependencies": { + "inherits": "^2.0.1", + "readable-stream": "^2.0.2" + } + }, + "node_modules/string_decoder": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", + "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", + "dependencies": { + "safe-buffer": "~5.2.0" + } + }, "node_modules/string-length": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/string-length/-/string-length-4.0.2.tgz", @@ -5301,6 +7666,27 @@ "node": ">=8" } }, + "node_modules/strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "dependencies": { + "ansi-regex": "^4.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/strip-ansi/node_modules/ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", + "dev": true, + "engines": { + "node": ">=6" + } + }, "node_modules/strip-bom": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-4.0.0.tgz", @@ -5336,6 +7722,14 @@ "resolved": "https://registry.npmjs.org/stylis/-/stylis-4.3.2.tgz", "integrity": "sha512-bhtUjWd/z6ltJiQwg0dUfxEJ+W+jdqQd8TbWLWyeIJHlnsqmGLRFFd8e5mA0AZi/zx90smXRlN66YMTcaSFifg==" }, + "node_modules/subarg": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/subarg/-/subarg-1.0.0.tgz", + "integrity": "sha512-RIrIdRY0X1xojthNcVtgT9sjpOGagEUKpZdgBUi054OEPFo282yg+zE+t1Rj3+RqKq2xStL7uUHhY+AjbC4BXg==", + "dependencies": { + "minimist": "^1.1.0" + } + }, "node_modules/supports-color": { "version": "7.2.0", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", @@ -5352,7 +7746,6 @@ "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, "engines": { "node": ">= 0.4" }, @@ -5360,6 +7753,58 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/syntax-error": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/syntax-error/-/syntax-error-1.4.0.tgz", + "integrity": "sha512-YPPlu67mdnHGTup2A8ff7BC2Pjq0e0Yp/IyTFN03zWO0RcK07uLcbi7C2KpGR2FvWbaB0+bfE27a+sBKebSo7w==", + "dependencies": { + "acorn-node": "^1.2.0" + } + }, + "node_modules/table": { + "version": "5.4.6", + "resolved": "https://registry.npmjs.org/table/-/table-5.4.6.tgz", + "integrity": "sha512-wmEc8m4fjnob4gt5riFRtTu/6+4rSe12TpAELNSqHMfF3IqnA+CH37USM6/YR3qRZv7e56kAEAtd6nKZaxe0Ug==", + "dev": true, + "dependencies": { + "ajv": "^6.10.2", + "lodash": "^4.17.14", + "slice-ansi": "^2.1.0", + "string-width": "^3.0.0" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/table/node_modules/emoji-regex": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", + "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", + "dev": true + }, + "node_modules/table/node_modules/is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha512-VHskAKYM8RfSFXwee5t5cbN5PZeq1Wrh6qd5bkyiXIf6UQcN6w/A0eXM9r6t8d+GYOh+o6ZhiEnb88LN/Y8m2w==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/table/node_modules/string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "dev": true, + "dependencies": { + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" + }, + "engines": { + "node": ">=6" + } + }, "node_modules/test-exclude": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/test-exclude/-/test-exclude-6.0.0.tgz", @@ -5374,6 +7819,49 @@ "node": ">=8" } }, + "node_modules/text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw==", + "dev": true + }, + "node_modules/through": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", + "integrity": "sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg==" + }, + "node_modules/through2": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/through2/-/through2-2.0.5.tgz", + "integrity": "sha512-/mrRod8xqpA+IHSLyGCQ2s8SPHiCDEeQJSep1jqLYeEUClOFG2Qsh+4FU6G9VeqpZnGW/Su8LQGc4YKni5rYSQ==", + "dependencies": { + "readable-stream": "~2.3.6", + "xtend": "~4.0.1" + } + }, + "node_modules/timers-browserify": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/timers-browserify/-/timers-browserify-1.4.2.tgz", + "integrity": "sha512-PIxwAupJZiYU4JmVZYwXp9FKsHMXb5h0ZEFyuXTAn8WLHOlcij+FEcbrvDsom1o5dr1YggEtFbECvGCW2sT53Q==", + "dependencies": { + "process": "~0.11.0" + }, + "engines": { + "node": ">=0.6.0" + } + }, + "node_modules/tmp": { + "version": "0.0.33", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", + "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", + "dev": true, + "dependencies": { + "os-tmpdir": "~1.0.2" + }, + "engines": { + "node": ">=0.6.0" + } + }, "node_modules/tmpl": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/tmpl/-/tmpl-1.0.5.tgz", @@ -5538,6 +8026,113 @@ "node": ">=0.3.1" } }, + "node_modules/tsconfig": { + "version": "5.0.3", + "resolved": "https://registry.npmjs.org/tsconfig/-/tsconfig-5.0.3.tgz", + "integrity": "sha512-Cq65A3kVp6BbsUgg9DRHafaGmbMb9EhAc7fjWvudNWKjkbWrt43FnrtZt6awshH1R0ocfF2Z0uxock3lVqEgOg==", + "dependencies": { + "any-promise": "^1.3.0", + "parse-json": "^2.2.0", + "strip-bom": "^2.0.0", + "strip-json-comments": "^2.0.0" + } + }, + "node_modules/tsconfig/node_modules/parse-json": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz", + "integrity": "sha512-QR/GGaKCkhwk1ePQNYDRKYZ3mwU9ypsKhB0XyFnLQdomyEqk3e8wpW3V5Jp88zbxK4n5ST1nqo+g9juTpownhQ==", + "dependencies": { + "error-ex": "^1.2.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/tsconfig/node_modules/strip-bom": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-2.0.0.tgz", + "integrity": "sha512-kwrX1y7czp1E69n2ajbG65mIo9dqvJ+8aBQXOGVxqwvNbsXdFM6Lq37dLAY3mknUwru8CfcCbfOLL/gMo+fi3g==", + "dependencies": { + "is-utf8": "^0.2.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/tsconfig/node_modules/strip-json-comments": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", + "integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/tsify": { + "version": "5.0.4", + "resolved": "https://registry.npmjs.org/tsify/-/tsify-5.0.4.tgz", + "integrity": "sha512-XAZtQ5OMPsJFclkZ9xMZWkSNyMhMxEPsz3D2zu79yoKorH9j/DT4xCloJeXk5+cDhosEibu4bseMVjyPOAyLJA==", + "dependencies": { + "convert-source-map": "^1.1.0", + "fs.realpath": "^1.0.0", + "object-assign": "^4.1.0", + "semver": "^6.1.0", + "through2": "^2.0.0", + "tsconfig": "^5.0.3" + }, + "engines": { + "node": ">=0.12" + }, + "peerDependencies": { + "browserify": ">= 10.x", + "typescript": ">= 2.8" + } + }, + "node_modules/tsify/node_modules/semver": { + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + }, + "node_modules/tsutils": { + "version": "3.21.0", + "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-3.21.0.tgz", + "integrity": "sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==", + "dev": true, + "dependencies": { + "tslib": "^1.8.1" + }, + "engines": { + "node": ">= 6" + }, + "peerDependencies": { + "typescript": ">=2.8.0 || >= 3.2.0-dev || >= 3.3.0-dev || >= 3.4.0-dev || >= 3.5.0-dev || >= 3.6.0-dev || >= 3.6.0-beta || >= 3.7.0-dev || >= 3.7.0-beta" + } + }, + "node_modules/tty-browserify": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/tty-browserify/-/tty-browserify-0.0.1.tgz", + "integrity": "sha512-C3TaO7K81YvjCgQH9Q1S3R3P3BtN3RIM8n+OvX4il1K1zgE8ZhI0op7kClgkxtutIE8hQrcrHBXvIheqKUUCxw==" + }, + "node_modules/type-check": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", + "integrity": "sha512-ZCmOJdvOWDBYJlzAoFkC+Q0+bUyEOS1ltgp1MGU03fqHG+dbi9tBFU2Rd9QKiDZFAYrhPh2JUf7rZRIuHRKtOg==", + "dev": true, + "dependencies": { + "prelude-ls": "~1.1.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, "node_modules/type-detect": { "version": "4.0.8", "resolved": "https://registry.npmjs.org/type-detect/-/type-detect-4.0.8.tgz", @@ -5547,11 +8142,24 @@ "node": ">=4" } }, + "node_modules/type-fest": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", + "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/typedarray": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", + "integrity": "sha512-/aCDEGatGvZ2BIk+HmLf4ifCJFwvKFNb9/JeZPMulfgFracn9QFcAf5GO8B/mweUjSoblS5In0cWhqpfs/5PQA==" + }, "node_modules/typescript": { "version": "5.5.2", "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.5.2.tgz", "integrity": "sha512-NcRtPEOsPFFWjobJEtfihkLCZCXZt/os3zf8nTxjVH3RvTSxjrCamJpbExGvYOF+tFHc3pA65qpdwPbzjohhew==", - "dev": true, "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" @@ -5560,6 +8168,29 @@ "node": ">=14.17" } }, + "node_modules/umd": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/umd/-/umd-3.0.3.tgz", + "integrity": "sha512-4IcGSufhFshvLNcMCV80UnQVlZ5pMOC8mvNPForqwA4+lzYQuetTESLDQkeLmihq8bRcnpbQa48Wb8Lh16/xow==", + "bin": { + "umd": "bin/cli.js" + } + }, + "node_modules/undeclared-identifiers": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/undeclared-identifiers/-/undeclared-identifiers-1.1.3.tgz", + "integrity": "sha512-pJOW4nxjlmfwKApE4zvxLScM/njmwj/DiUBv7EabwE4O8kRUy+HIwxQtZLBPll/jx1LJyBcqNfB3/cpv9EZwOw==", + "dependencies": { + "acorn-node": "^1.3.0", + "dash-ast": "^1.0.0", + "get-assigned-identifiers": "^1.2.0", + "simple-concat": "^1.0.0", + "xtend": "^4.0.1" + }, + "bin": { + "undeclared-identifiers": "bin.js" + } + }, "node_modules/underscore": { "version": "1.1.7", "resolved": "https://registry.npmjs.org/underscore/-/underscore-1.1.7.tgz", @@ -5610,6 +8241,33 @@ "browserslist": ">= 4.21.0" } }, + "node_modules/uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dev": true, + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/uri-js/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/url": { + "version": "0.11.3", + "resolved": "https://registry.npmjs.org/url/-/url-0.11.3.tgz", + "integrity": "sha512-6hxOLGfZASQK/cijlZnZJTq8OXAkt/3YGfQX45vvMYXpZoo8NdWZcY73K108Jf759lS1Bv/8wXnHDTSz17dSRw==", + "dependencies": { + "punycode": "^1.4.1", + "qs": "^6.11.2" + } + }, "node_modules/util": { "version": "0.10.4", "resolved": "https://registry.npmjs.org/util/-/util-0.10.4.tgz", @@ -5618,6 +8276,11 @@ "inherits": "2.0.3" } }, + "node_modules/util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==" + }, "node_modules/uuid": { "version": "9.0.1", "resolved": "https://registry.npmjs.org/uuid/-/uuid-9.0.1.tgz", @@ -5647,6 +8310,12 @@ "node": ">=8" } }, + "node_modules/v8-compile-cache": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.4.0.tgz", + "integrity": "sha512-ocyWc3bAHBB/guyqJQVI5o4BZkPhznPYUG2ea80Gond/BgNWpap8TOmLSeeQG7bnh2KMISxskdADG59j7zruhw==", + "dev": true + }, "node_modules/v8-compile-cache-lib": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz", @@ -5673,6 +8342,11 @@ "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==", "dev": true }, + "node_modules/vm-browserify": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vm-browserify/-/vm-browserify-1.1.2.tgz", + "integrity": "sha512-2ham8XPWTONajOR0ohOKOHXkm3+gaBmGut3SRuu75xLd/RRaY6vqgh8NBYYk7+RW3u5AtzPQZG8F10LHkl0lAQ==" + }, "node_modules/walker": { "version": "1.0.8", "resolved": "https://registry.npmjs.org/walker/-/walker-1.0.8.tgz", @@ -5687,6 +8361,45 @@ "resolved": "https://registry.npmjs.org/web-worker/-/web-worker-1.3.0.tgz", "integrity": "sha512-BSR9wyRsy/KOValMgd5kMyr3JzpdeoR9KVId8u5GVlTTAtNChlsE4yTxeY7zMdNSyOmoKBv8NH2qeRY9Tg+IaA==" }, + "node_modules/which": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", + "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "which": "bin/which" + } + }, + "node_modules/which-typed-array": { + "version": "1.1.15", + "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.15.tgz", + "integrity": "sha512-oV0jmFtUky6CXfkqehVvBP/LSWJ2sy4vWMioiENyJLePrBO/yKyV9OyJySfAKosh+RYkIl5zJCNZ8/4JncrpdA==", + "dependencies": { + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "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", + "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/wrap-ansi": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", @@ -5719,8 +8432,19 @@ "node_modules/wrappy": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", - "dev": true + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==" + }, + "node_modules/write": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/write/-/write-1.0.3.tgz", + "integrity": "sha512-/lg70HAjtkUgWPVZhZcm+T4hkL8Zbtp1nFNOn3lRrxnlv50SRBv7cR7RqR+GMsd3hUXy9hWBo4CHTbFTcOYwig==", + "dev": true, + "dependencies": { + "mkdirp": "^0.5.1" + }, + "engines": { + "node": ">=4" + } }, "node_modules/write-file-atomic": { "version": "4.0.2", @@ -5741,6 +8465,14 @@ "integrity": "sha512-huCv9IH9Tcf95zuYCsQraZtWnJvBtLVE0QHMOs8bWyZAFZNDcYjsPq1nEx8jKA9y+Beo9v+7OBPRisQTjinQMw==", "dev": true }, + "node_modules/xtend": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", + "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", + "engines": { + "node": ">=0.4" + } + }, "node_modules/y18n": { "version": "5.0.8", "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", diff --git a/package.json b/package.json index 39a4897..c623db7 100644 --- a/package.json +++ b/package.json @@ -17,7 +17,7 @@ "build": "tsc && npm run build:types", "build:types": "tsc -p tsconfig-types.json", "build:help": "tsc --help", - "build:client": "npm run mkdir:dist && browserify src/index.ts -p [ tsify --noImplicitAny ] > dist/littleMermaid.js", + "build:client": "npm run mkdir:dist && browserify src/sql.ts -p [ tsify --noImplicitAny ] > dist/sql.js", "mkdir:dist": "node ./build/createDistFolder.js" }, "repository": { @@ -37,10 +37,16 @@ "homepage": "https://github.com/lastlink/sqltooling-drawio#readme", "dependencies": { "@funktechno/little-mermaid-2-the-sql": "^0.1.1", - "@funktechno/sqlsimpleparser": "^0.1.0" + "@funktechno/sqlsimpleparser": "^0.1.0", + "browserify": "^17.0.0", + "tsify": "^5.0.2" }, "devDependencies": { + "@types/eslint": "^7.2.4", "@types/jest": "^29.0.3", + "@typescript-eslint/eslint-plugin": "^2.24.0", + "@typescript-eslint/parser": "^2.24.0", + "eslint": "^6.8.0", "jest": "^29.0.3", "jest-junit": "^16.0.0", "jest-serial-runner": "^1.1.0", diff --git a/src/sql.ts b/src/sql.ts index de6eb9e..1f69623 100644 --- a/src/sql.ts +++ b/src/sql.ts @@ -18,28 +18,28 @@ Draw.loadPlugin(function(ui) { // DbParser //Create Base div - const divGenSQL = document.createElement('div'); - divGenSQL.style.userSelect = 'none'; - divGenSQL.style.overflow = 'hidden'; - divGenSQL.style.padding = '10px'; - divGenSQL.style.height = '100%'; - - const sqlInputGenSQL = document.createElement('textarea'); - sqlInputGenSQL.style.height = '200px'; - sqlInputGenSQL.style.width = '100%'; - sqlInputGenSQL.value = '-- click a database type button' + const divGenSQL = document.createElement("div"); + divGenSQL.style.userSelect = "none"; + divGenSQL.style.overflow = "hidden"; + divGenSQL.style.padding = "10px"; + divGenSQL.style.height = "100%"; + + const sqlInputGenSQL = document.createElement("textarea"); + sqlInputGenSQL.style.height = "200px"; + sqlInputGenSQL.style.width = "100%"; + sqlInputGenSQL.value = "-- click a database type button"; mxUtils.br(divGenSQL); divGenSQL.appendChild(sqlInputGenSQL); - var theMenuExportAs = ui.menus.get('exportAs'); - let buttonLabel = 'tosql=To SQL' + const theMenuExportAs = ui.menus.get("exportAs"); + let buttonLabel = "tosql=To SQL"; // vscode extension support if(!(theMenuExportAs && theMenuExportAs.enabled)) { - buttonLabel = 'tosql=Export As SQL' + buttonLabel = "tosql=Export As SQL"; } // Extends Extras menu mxResources.parse(buttonLabel); - const wndGenSQL = new mxWindow(mxResources.get('tosql'), divGenSQL, document.body.offsetWidth - 480, 140, + const wndGenSQL = new mxWindow(mxResources.get("tosql"), divGenSQL, document.body.offsetWidth - 480, 140, 320, 320, true, true); wndGenSQL.destroyOnClose = false; wndGenSQL.setMaximizable(false); @@ -50,10 +50,10 @@ Draw.loadPlugin(function(ui) { * return text quantifiers for dialect * @returns json */ - function GetColumnQuantifiers(type: "mysql" | "sqlserver" | "sqlite" | "postgres" | undefined):ColumnQuantifiers { - let chars = { - Start: '"', - End: '"', + function GetColumnQuantifiers(type: "mysql" | "sqlserver" | "sqlite" | "postgres" | undefined): ColumnQuantifiers { + const chars = { + Start: "\"", + End: "\"", }; if (type == "mysql") { chars.Start = "`"; @@ -70,10 +70,10 @@ Draw.loadPlugin(function(ui) { * @param {*} label * @returns */ - function removeHtml(label:string){ - var div = document.createElement("div"); + function removeHtml(label: string){ + const div = document.createElement("div"); divGenSQL.innerHTML = label; - var text = div.textContent || div.innerText || ""; + const text = div.textContent || div.innerText || ""; return text; } /** @@ -82,43 +82,43 @@ Draw.loadPlugin(function(ui) { * @param {*} columnQuantifiers * @returns */ - function getDbLabel(label:string, columnQuantifiers:ColumnQuantifiers): TableAttribute{ - label = removeHtml(label) + function getDbLabel(label: string, columnQuantifiers: ColumnQuantifiers): TableAttribute{ + label = removeHtml(label); // fix duplicate spaces and different space chars label = label - .replace(/\s+/g, " ") - let firstSpaceIndex = label[0] == columnQuantifiers.Start && + .replace(/\s+/g, " "); + const firstSpaceIndex = label[0] == columnQuantifiers.Start && label.indexOf(columnQuantifiers.End + " ") !== -1 ? label.indexOf(columnQuantifiers.End + " ") : label.indexOf(" "); - let attributeType = label.substring(firstSpaceIndex + 1).trim(); - let attributeName = RemoveNameQuantifiers(label.substring(0, firstSpaceIndex)); - let attribute = { + const attributeType = label.substring(firstSpaceIndex + 1).trim(); + const attributeName = RemoveNameQuantifiers(label.substring(0, firstSpaceIndex)); + const attribute = { attributeName, attributeType - } - return attribute + }; + return attribute; } - function RemoveNameQuantifiers(name:string) { + function RemoveNameQuantifiers(name: string) { return name.replace(/\[|\]|\(|\"|\'|\`/g, "").trim(); } - function getMermaidDiagramDb(type: "mysql" | "sqlserver" | "sqlite" | "postgres" | undefined):DbDefinition{ - var model = ui.editor.graph.getModel() + function getMermaidDiagramDb(type: "mysql" | "sqlserver" | "sqlite" | "postgres" | undefined): DbDefinition{ + const model = ui.editor.graph.getModel(); // same models from mermaid for diagram relationships // only difference is entities is an array rather than object to allow duplicate tables - let entities: Record = {} - let relationships:DbRelationshipDefinition[] = [] + const entities: Record = {}; + const relationships: DbRelationshipDefinition[] = []; // build models for (const key in model.cells) { if (Object.hasOwnProperty.call(model.cells, key)) { const mxcell = model.cells[key]; if(mxcell.mxObjectId.indexOf("mxCell") !== -1) { if(mxcell.style && mxcell.style.trim().startsWith("swimlane;")){ - let entity:TableEntity = { + const entity: TableEntity = { name: RemoveNameQuantifiers(mxcell.value), attributes: [] as TableAttribute[], - } + }; for (let c = 0; c < mxcell.children.length; c++) { const col = mxcell.children[c]; if(col.mxObjectId.indexOf("mxCell") !== -1) { @@ -126,16 +126,16 @@ Draw.loadPlugin(function(ui) { const columnQuantifiers = GetColumnQuantifiers(type); //Get delimiter of column name //Get full name - let attribute = getDbLabel(col.value, columnQuantifiers) - var attributeKeyType = col.children.find(x=> ["FK","PK"].findIndex(k => k== x.value.toUpperCase()) !== -1 || - x.value.toUpperCase().indexOf("PK,")!=-1) + const attribute = getDbLabel(col.value, columnQuantifiers); + const attributeKeyType = col.children.find(x=> ["FK","PK"].findIndex(k => k== x.value.toUpperCase()) !== -1 || + x.value.toUpperCase().indexOf("PK,")!=-1); if(attributeKeyType){ - attribute.attributeKeyType = attributeKeyType.value + attribute.attributeKeyType = attributeKeyType.value; if(attribute.attributeKeyType != "PK" && attribute.attributeKeyType.indexOf("PK") != -1){ - attribute.attributeKeyType = "PK" + attribute.attributeKeyType = "PK"; } } - entity.attributes.push(attribute) + entity.attributes.push(attribute); if(col.edges && col.edges.length){ // check for edges foreign keys for (let e = 0; e < col.edges.length; e++) { @@ -147,19 +147,19 @@ Draw.loadPlugin(function(ui) { // extract endArrow txt // check if both match and contain many or open // if both match and are many then create a new table - let endCheck = "endArrow=" - let endArr = edge.style.indexOf(endCheck) != -1 ? + const endCheck = "endArrow="; + const endArr = edge.style.indexOf(endCheck) != -1 ? edge.style.substring(edge.style.indexOf(endCheck) + endCheck.length, edge.style.substring(edge.style.indexOf(endCheck) + endCheck.length).indexOf(";") + edge.style.indexOf(endCheck) + endCheck.length) - : "" - let startCheck = "startArrow=" - let startArr = edge.style.indexOf(startCheck) != -1 ? + : ""; + const startCheck = "startArrow="; + const startArr = edge.style.indexOf(startCheck) != -1 ? edge.style.substring(edge.style.indexOf(startCheck) + startCheck.length, edge.style.substring(edge.style.indexOf(startCheck) + startCheck.length).indexOf(";") + edge.style.indexOf(startCheck) + startCheck.length) - : "" + : ""; - var manyCheck = ["open","many"] - var sourceIsPrimary = endArr && manyCheck + const manyCheck = ["open","many"]; + const sourceIsPrimary = endArr && manyCheck .findIndex(x => endArr.toLocaleLowerCase().indexOf(x)!=-1) != -1; - var targetIsPrimary = startArr && manyCheck + const targetIsPrimary = startArr && manyCheck .findIndex(x => startArr.toLocaleLowerCase().indexOf(x)!=-1) != -1; // has to be one to many and not one to one if((targetIsPrimary || sourceIsPrimary) && @@ -167,87 +167,87 @@ Draw.loadPlugin(function(ui) { ){ var sourceId = edge.source.value; var sourceAttr = getDbLabel(sourceId, columnQuantifiers); - sourceId = sourceAttr.attributeName + sourceId = sourceAttr.attributeName; var sourceEntity = RemoveNameQuantifiers(edge.source.parent.value); var targetId = edge.target.value; var targetAttr = getDbLabel(targetId, columnQuantifiers); - targetId = targetAttr.attributeName + targetId = targetAttr.attributeName; var targetEntity = RemoveNameQuantifiers(edge.target.parent.value); // entityA primary // entityB foreign - let relationship:DbRelationshipDefinition = { + const relationship: DbRelationshipDefinition = { entityA: sourceIsPrimary ? sourceEntity : targetEntity, entityB: sourceIsPrimary ? targetEntity : sourceEntity, // based off of styles? relSpec: { - cardA: 'ZERO_OR_MORE', - cardB: 'ONLY_ONE', + cardA: "ZERO_OR_MORE", + cardB: "ONLY_ONE", relType: "IDENTIFYING" }, roleA: sourceIsPrimary ? `[${sourceEntity}.${sourceId}] to [${targetEntity}.${targetId}]` : `[${targetEntity}.${targetId}] to [${sourceEntity}.${sourceId}]` - } + }; // check that is doesn't already exist - var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA) + var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA); if(exists ==-1){ - relationships.push(relationship) + relationships.push(relationship); } } else if(targetIsPrimary && sourceIsPrimary){ // add a new many to many table var sourceId = edge.source.value; sourceAttr = getDbLabel(sourceId, columnQuantifiers); - sourceAttr.attributeKeyType = "PK" - sourceId = sourceAttr.attributeName + sourceAttr.attributeKeyType = "PK"; + sourceId = sourceAttr.attributeName; var sourceEntity = RemoveNameQuantifiers(edge.source.parent.value); var targetId = edge.target.value; targetAttr = getDbLabel(targetId, columnQuantifiers); - targetAttr.attributeKeyType = "PK" - targetId = targetAttr.attributeName + targetAttr.attributeKeyType = "PK"; + targetId = targetAttr.attributeName; var targetEntity = RemoveNameQuantifiers(edge.target.parent.value); - let compositeEntity = { + const compositeEntity = { name: RemoveNameQuantifiers(sourceEntity) + "_" + RemoveNameQuantifiers(targetEntity), attributes: [sourceAttr, targetAttr] - } + }; // add composite entity if(entities[compositeEntity.name]){ // DON'T add duplicate composite tables } else { - entities[compositeEntity.name] = compositeEntity + entities[compositeEntity.name] = compositeEntity; } // entityA primary // entityB foreign - let relationship = { + const relationship = { entityA: sourceEntity, entityB: compositeEntity.name, // based off of styles? relSpec: { - cardA: 'ZERO_OR_MORE', - cardB: 'ONLY_ONE', + cardA: "ZERO_OR_MORE", + cardB: "ONLY_ONE", relType: "IDENTIFYING" }, roleA: `[${sourceEntity}.${sourceId}] to [${compositeEntity.name}.${sourceId}]` - } + }; // check that is doesn't already exist - var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA) + var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA); if(exists ==-1){ - relationships.push(relationship) + relationships.push(relationship); } - let relationship2 = { + const relationship2 = { entityA: targetEntity, entityB: compositeEntity.name, // based off of styles? relSpec: { - cardA: 'ZERO_OR_MORE', - cardB: 'ONLY_ONE', + cardA: "ZERO_OR_MORE", + cardB: "ONLY_ONE", relType: "IDENTIFYING" }, roleA: `[${targetEntity}.${targetId}] to [${compositeEntity.name}.${targetId}]` - } + }; // check that is doesn't already exist - exists = relationships.findIndex(r => r.entityA == relationship2.entityA && r.entityB == relationship2.entityB && r.roleA == relationship2.roleA) + exists = relationships.findIndex(r => r.entityA == relationship2.entityA && r.entityB == relationship2.entityB && r.roleA == relationship2.roleA); if(exists ==-1){ - relationships.push(relationship2) + relationships.push(relationship2); } } @@ -261,13 +261,13 @@ Draw.loadPlugin(function(ui) { } // allows for duplicates if another table has the same name if(entities[entity.name]){ - var count = 2; + let count = 2; while(entities[entity.name + count.toString()]){ count++; } - entities[entity.name + count.toString()] = entity + entities[entity.name + count.toString()] = entity; } else { - entities[entity.name] = entity + entities[entity.name] = entity; } } @@ -276,7 +276,7 @@ Draw.loadPlugin(function(ui) { } class DatabaseModel{ - constructor(entities: Record, relationships:DbRelationshipDefinition[]){ + constructor(entities: Record, relationships: DbRelationshipDefinition[]){ this.entities = entities; this.relationships = relationships; } @@ -286,15 +286,15 @@ Draw.loadPlugin(function(ui) { private relationships: DbRelationshipDefinition[]; getEntities(){ - return this.entities + return this.entities; } getRelationships(){ - return this.relationships + return this.relationships; } } - var db = new DatabaseModel(entities, relationships) as unknown as DbDefinition; + const db = new DatabaseModel(entities, relationships) as unknown as DbDefinition; return db; } @@ -302,64 +302,64 @@ Draw.loadPlugin(function(ui) { function generateSql(type: "mysql" | "sqlserver" | "sqlite" | "postgres" | undefined) { // get diagram model - var db = getMermaidDiagramDb(type); + const db = getMermaidDiagramDb(type); // load parser - var parser = new DbParser(type as string, db) + const parser = new DbParser(type as string, db); // generate sql - var sql = parser.getSQLDataDefinition() - sql = `/*\n\tGenerated in drawio\n\tDatabase: ${type}\n*/\n\n` + sql + let sql = parser.getSQLDataDefinition(); + sql = `/*\n\tGenerated in drawio\n\tDatabase: ${type}\n*/\n\n` + sql; sql = sql.trim(); // update sql value in text area sqlInputGenSQL.value = sql; // TODO: use selection as well? - var modelSelected = ui.editor.graph.getSelectionModel() + const modelSelected = ui.editor.graph.getSelectionModel(); }; mxUtils.br(divGenSQL); - const resetBtnGenSQL = mxUtils.button(mxResources.get('reset'), function() { - sqlInputGenSQL.value = ''; + const resetBtnGenSQL = mxUtils.button(mxResources.get("reset"), function() { + sqlInputGenSQL.value = ""; }); - resetBtnGenSQL.style.marginTop = '8px'; - resetBtnGenSQL.style.marginRight = '4px'; - resetBtnGenSQL.style.padding = '4px'; + resetBtnGenSQL.style.marginTop = "8px"; + resetBtnGenSQL.style.marginRight = "4px"; + resetBtnGenSQL.style.padding = "4px"; divGenSQL.appendChild(resetBtnGenSQL); - const btnGenSQL_mysql = mxUtils.button('MySQL', function() { - generateSql('mysql'); + const btnGenSQL_mysql = mxUtils.button("MySQL", function() { + generateSql("mysql"); }); - btnGenSQL_mysql.style.marginTop = '8px'; - btnGenSQL_mysql.style.padding = '4px'; + btnGenSQL_mysql.style.marginTop = "8px"; + btnGenSQL_mysql.style.padding = "4px"; divGenSQL.appendChild(btnGenSQL_mysql); - const btnGenSQL_sqlserver = mxUtils.button('SQL Server', function() { - generateSql('sqlserver'); + const btnGenSQL_sqlserver = mxUtils.button("SQL Server", function() { + generateSql("sqlserver"); }); - btnGenSQL_sqlserver.style.marginTop = '8px'; - btnGenSQL_sqlserver.style.padding = '4px'; + btnGenSQL_sqlserver.style.marginTop = "8px"; + btnGenSQL_sqlserver.style.padding = "4px"; divGenSQL.appendChild(btnGenSQL_sqlserver); - const btnGenSQL_postgres = mxUtils.button('PostgreSQL', function() { - generateSql('postgres'); + const btnGenSQL_postgres = mxUtils.button("PostgreSQL", function() { + generateSql("postgres"); }); - btnGenSQL_postgres.style.marginTop = '8px'; - btnGenSQL_postgres.style.padding = '4px'; + btnGenSQL_postgres.style.marginTop = "8px"; + btnGenSQL_postgres.style.padding = "4px"; divGenSQL.appendChild(btnGenSQL_postgres); - const btnGenSQL_sqlite = mxUtils.button('Sqlite', function() { - generateSql('sqlite'); + const btnGenSQL_sqlite = mxUtils.button("Sqlite", function() { + generateSql("sqlite"); }); - btnGenSQL_sqlite.style.marginTop = '8px'; - btnGenSQL_sqlite.style.padding = '4px'; + btnGenSQL_sqlite.style.marginTop = "8px"; + btnGenSQL_sqlite.style.padding = "4px"; divGenSQL.appendChild(btnGenSQL_sqlite); // Adds action - ui.actions.addAction('tosql', function() { + ui.actions.addAction("tosql", function() { wndGenSQL.setVisible(!wndGenSQL.isVisible()); if (wndGenSQL.isVisible()) { @@ -372,66 +372,66 @@ Draw.loadPlugin(function(ui) { //Table Info - var foreignKeyList:ForeignKeyModel[] = []; - var primaryKeyList:PrimaryKeyModel[] = []; - var tableList:TableModel[] = []; - var cells:mxCell[] = []; - var tableCell:mxCell|null = null; - var rowCell:mxCell|null = null; - var dx = 0; - var exportedTables = 0; + let foreignKeyList: ForeignKeyModel[] = []; + let primaryKeyList: PrimaryKeyModel[] = []; + let tableList: TableModel[] = []; + let cells: mxCell[] = []; + let tableCell: mxCell|null = null; + let rowCell: mxCell|null = null; + let dx = 0; + let exportedTables = 0; //Create Base div - const divFromSQL = document.createElement('div'); - divFromSQL.style.userSelect = 'none'; - divFromSQL.style.overflow = 'hidden'; - divFromSQL.style.padding = '10px'; - divFromSQL.style.height = '100%'; + const divFromSQL = document.createElement("div"); + divFromSQL.style.userSelect = "none"; + divFromSQL.style.overflow = "hidden"; + divFromSQL.style.padding = "10px"; + divFromSQL.style.height = "100%"; var graph = ui.editor.graph; - const sqlInputFromSQL = document.createElement('textarea'); - sqlInputFromSQL.style.height = '200px'; - sqlInputFromSQL.style.width = '100%'; - var defaultReset = '/*\n\tDraw io default value\n*/\n\nCREATE TABLE Persons\n(\n PersonID int NOT NULL,\n LastName varchar(255),\n ' + - 'FirstName varchar(255),\n Address varchar(255),\n City varchar(255),\n Primary Key(PersonID)\n);\n\n' + - 'CREATE TABLE Orders\n(\n OrderID int NOT NULL PRIMARY KEY,\n PersonID int NOT NULL,\n FOREIGN KEY ([PersonID]) REFERENCES [Persons]([PersonID])' + - '\n);' - sqlInputFromSQL.value = defaultReset + const sqlInputFromSQL = document.createElement("textarea"); + sqlInputFromSQL.style.height = "200px"; + sqlInputFromSQL.style.width = "100%"; + const defaultReset = "/*\n\tDraw io default value\n*/\n\nCREATE TABLE Persons\n(\n PersonID int NOT NULL,\n LastName varchar(255),\n " + + "FirstName varchar(255),\n Address varchar(255),\n City varchar(255),\n Primary Key(PersonID)\n);\n\n" + + "CREATE TABLE Orders\n(\n OrderID int NOT NULL PRIMARY KEY,\n PersonID int NOT NULL,\n FOREIGN KEY ([PersonID]) REFERENCES [Persons]([PersonID])" + + "\n);"; + sqlInputFromSQL.value = defaultReset; mxUtils.br(divFromSQL); divFromSQL.appendChild(sqlInputFromSQL); var graph = ui.editor.graph; // Extends Extras menu - mxResources.parse('fromSql=From SQL'); + mxResources.parse("fromSql=From SQL"); - const wndFromSQL = new mxWindow(mxResources.get('fromSql'), divFromSQL, document.body.offsetWidth - 480, 140, + const wndFromSQL = new mxWindow(mxResources.get("fromSql"), divFromSQL, document.body.offsetWidth - 480, 140, 320, 320, true, true); wndFromSQL.destroyOnClose = false; wndFromSQL.setMaximizable(false); wndFromSQL.setResizable(false); wndFromSQL.setClosable(true); - function AddRow(propertyModel:PropertyModel, tableName:string) { + function AddRow(propertyModel: PropertyModel, tableName: string) { - var cellName = propertyModel.Name + (propertyModel.ColumnProperties ? " " + propertyModel.ColumnProperties: ""); + const cellName = propertyModel.Name + (propertyModel.ColumnProperties ? " " + propertyModel.ColumnProperties: ""); rowCell = new mxCell(cellName, new mxGeometry(0, 0, 90, 26), - 'shape=partialRectangle;top=0;left=0;right=0;bottom=0;align=left;verticalAlign=top;spacingTop=-2;fillColor=none;spacingLeft=64;spacingRight=4;overflow=hidden;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;dropTarget=0;'); + "shape=partialRectangle;top=0;left=0;right=0;bottom=0;align=left;verticalAlign=top;spacingTop=-2;fillColor=none;spacingLeft=64;spacingRight=4;overflow=hidden;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;dropTarget=0;"); rowCell.vertex = true; - var columnType = propertyModel.IsPrimaryKey && propertyModel.IsForeignKey ? 'PK | FK' : propertyModel.IsPrimaryKey ? 'PK' : propertyModel.IsForeignKey ? 'FK' : ''; + const columnType = propertyModel.IsPrimaryKey && propertyModel.IsForeignKey ? "PK | FK" : propertyModel.IsPrimaryKey ? "PK" : propertyModel.IsForeignKey ? "FK" : ""; - var left = sb.cloneCell(rowCell, columnType); + const left = sb.cloneCell(rowCell, columnType); left.connectable = false; - left.style = 'shape=partialRectangle;top=0;left=0;bottom=0;fillColor=none;align=left;verticalAlign=middle;spacingLeft=4;spacingRight=4;overflow=hidden;rotatable=180;points=[];portConstraint=eastwest;part=1;' + left.style = "shape=partialRectangle;top=0;left=0;bottom=0;fillColor=none;align=left;verticalAlign=middle;spacingLeft=4;spacingRight=4;overflow=hidden;rotatable=180;points=[];portConstraint=eastwest;part=1;"; left.geometry.width = 54; left.geometry.height = 26; rowCell.insert(left); - var size = ui.editor.graph.getPreferredSizeForCell(rowCell); + const size = ui.editor.graph.getPreferredSizeForCell(rowCell); if(tableCell){ if (size !== null && tableCell.geometry.width < size.width + 10) { @@ -446,9 +446,9 @@ Draw.loadPlugin(function(ui) { }; - function parseSql(text:string, type?: "mysql" | "sqlite" | "postgres" | "sqlserver" | undefined) { + function parseSql(text: string, type?: "mysql" | "sqlite" | "postgres" | "sqlserver" | undefined) { // reset values - cells = [] + cells = []; tableCell = null; rowCell = null; // load parser @@ -461,7 +461,6 @@ Draw.loadPlugin(function(ui) { .WithEnds() .ToModel(); - foreignKeyList = models.ForeignKeyList; primaryKeyList = models.PrimaryKeyList; @@ -475,16 +474,16 @@ Draw.loadPlugin(function(ui) { function CreateTableUI(type: "mysql" | "sqlite" | "postgres" | "sqlserver" | undefined) { tableList.forEach(function(tableModel) { //Define table size width - var maxNameLenght = 100 + tableModel.Name.length; + const maxNameLenght = 100 + tableModel.Name.length; //Create Table tableCell = new mxCell(tableModel.Name, new mxGeometry(dx, 0, maxNameLenght, 26), - 'swimlane;fontStyle=0;childLayout=stackLayout;horizontal=1;startSize=26;fillColor=default;horizontalStack=0;resizeParent=1;resizeLast=0;collapsible=1;marginBottom=0;swimlaneFillColor=default;align=center;'); + "swimlane;fontStyle=0;childLayout=stackLayout;horizontal=1;startSize=26;fillColor=default;horizontalStack=0;resizeParent=1;resizeLast=0;collapsible=1;marginBottom=0;swimlaneFillColor=default;align=center;"); tableCell.vertex = true; //Resize row if(rowCell){ - var size = ui.editor.graph.getPreferredSizeForCell(rowCell); + const size = ui.editor.graph.getPreferredSizeForCell(rowCell); if (size !== null) { tableCell.geometry.width = size.width + maxNameLenght; } @@ -506,33 +505,33 @@ Draw.loadPlugin(function(ui) { }); if (cells.length > 0) { - var graph = ui.editor.graph; - var view = graph.view; - var bds = graph.getGraphBounds(); + const graph = ui.editor.graph; + const view = graph.view; + const bds = graph.getGraphBounds(); // Computes unscaled, untranslated graph bounds - var x = Math.ceil(Math.max(0, bds.x / view.scale - view.translate.x) + 4 * graph.gridSize); - var y = Math.ceil(Math.max(0, (bds.y + bds.height) / view.scale - view.translate.y) + 4 * graph.gridSize); + const x = Math.ceil(Math.max(0, bds.x / view.scale - view.translate.x) + 4 * graph.gridSize); + const y = Math.ceil(Math.max(0, (bds.y + bds.height) / view.scale - view.translate.y) + 4 * graph.gridSize); graph.setSelectionCells(graph.importCells(cells, x, y)); // add foreign key edges - var model = graph.getModel(); + const model = graph.getModel(); const columnQuantifiers = GetColumnQuantifiers(type); - var pt = graph.getFreeInsertPoint(); + const pt = graph.getFreeInsertPoint(); foreignKeyList.forEach(function(fk){ if(fk.IsDestination && fk.PrimaryKeyName && fk.ReferencesPropertyName && fk.PrimaryKeyTableName && fk.ReferencesTableName) { - var insertEdge = mxUtils.bind(this, function(targetCell, sourceCell, edge){ - var label = "" - var edgeStyle = "edgeStyle=entityRelationEdgeStyle;html=1;endArrow=ERzeroToMany;startArrow=ERzeroToOne;labelBackgroundColor=none;fontFamily=Verdana;fontSize=14;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=-0.018;entryY=0.608;entryDx=0;entryDy=0;entryPerimeter=0;" - var edgeCell = graph.insertEdge(null, null, label || '', (edge.invert) ? + const insertEdge = mxUtils.bind(this, function(targetCell, sourceCell, edge){ + const label = ""; + const edgeStyle = "edgeStyle=entityRelationEdgeStyle;html=1;endArrow=ERzeroToMany;startArrow=ERzeroToOne;labelBackgroundColor=none;fontFamily=Verdana;fontSize=14;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=-0.018;entryY=0.608;entryDx=0;entryDy=0;entryPerimeter=0;"; + const edgeCell = graph.insertEdge(null, null, label || "", (edge.invert) ? sourceCell : targetCell, (edge.invert) ? targetCell : sourceCell, edgeStyle); }); - let edge = { + const edge = { invert: true }; - var targetCell = null; - var sourceCell = null; + let targetCell = null; + let sourceCell = null; // locate edge source and target cells for (const key in model.cells) { if(targetCell && sourceCell) @@ -540,12 +539,12 @@ Draw.loadPlugin(function(ui) { if (Object.hasOwnProperty.call(model.cells, key)) { const mxcell = model.cells[key]; if(mxcell.style && mxcell.style.trim().startsWith("swimlane;")){ - let entity = { + const entity = { name: mxcell.value, attributes: [] - } - var isPrimaryTable = entity.name == fk.PrimaryKeyTableName; - var isForeignTable = entity.name == fk.ReferencesTableName; + }; + const isPrimaryTable = entity.name == fk.PrimaryKeyTableName; + const isForeignTable = entity.name == fk.ReferencesTableName; if(isPrimaryTable || isForeignTable){ for (let c = 0; c < mxcell.children.length; c++) { if(targetCell && sourceCell) @@ -553,7 +552,7 @@ Draw.loadPlugin(function(ui) { const col = mxcell.children[c]; if(col.mxObjectId.indexOf("mxCell") !== -1) { if(col.style && col.style.trim().startsWith("shape=partialRectangle")){ - let attribute = getDbLabel(col.value, columnQuantifiers) + const attribute = getDbLabel(col.value, columnQuantifiers); if(isPrimaryTable && attribute.attributeName == fk.PrimaryKeyName){ targetCell = col; break; @@ -572,7 +571,7 @@ Draw.loadPlugin(function(ui) { if(targetCell && sourceCell) insertEdge(targetCell, sourceCell, edge); } - }) + }); graph.scrollCellToVisible(graph.getSelectionCell()); } @@ -581,49 +580,49 @@ Draw.loadPlugin(function(ui) { mxUtils.br(divFromSQL); - const resetBtnFromSQL = mxUtils.button(mxResources.get('reset'), function() { + const resetBtnFromSQL = mxUtils.button(mxResources.get("reset"), function() { sqlInputFromSQL.value = defaultReset; }); - resetBtnFromSQL.style.marginTop = '8px'; - resetBtnFromSQL.style.marginRight = '4px'; - resetBtnFromSQL.style.padding = '4px'; + resetBtnFromSQL.style.marginTop = "8px"; + resetBtnFromSQL.style.marginRight = "4px"; + resetBtnFromSQL.style.padding = "4px"; divFromSQL.appendChild(resetBtnFromSQL); - const btnFromSQL_mysql = mxUtils.button('Insert MySQL', function() { - parseSql(sqlInputFromSQL.value, 'mysql'); + const btnFromSQL_mysql = mxUtils.button("Insert MySQL", function() { + parseSql(sqlInputFromSQL.value, "mysql"); }); - btnFromSQL_mysql.style.marginTop = '8px'; - btnFromSQL_mysql.style.padding = '4px'; + btnFromSQL_mysql.style.marginTop = "8px"; + btnFromSQL_mysql.style.padding = "4px"; divFromSQL.appendChild(btnFromSQL_mysql); - const btnFromSQL_sqlserver = mxUtils.button('Insert SQL Server', function() { - parseSql(sqlInputFromSQL.value, 'sqlserver'); + const btnFromSQL_sqlserver = mxUtils.button("Insert SQL Server", function() { + parseSql(sqlInputFromSQL.value, "sqlserver"); }); - btnFromSQL_sqlserver.style.marginTop = '8px'; - btnFromSQL_sqlserver.style.padding = '4px'; + btnFromSQL_sqlserver.style.marginTop = "8px"; + btnFromSQL_sqlserver.style.padding = "4px"; divFromSQL.appendChild(btnFromSQL_sqlserver); - const btnFromSQL_postgres = mxUtils.button('Insert PostgreSQL', function() { - parseSql(sqlInputFromSQL.value, 'postgres'); + const btnFromSQL_postgres = mxUtils.button("Insert PostgreSQL", function() { + parseSql(sqlInputFromSQL.value, "postgres"); }); - btnFromSQL_postgres.style.marginTop = '8px'; - btnFromSQL_postgres.style.padding = '4px'; + btnFromSQL_postgres.style.marginTop = "8px"; + btnFromSQL_postgres.style.padding = "4px"; divFromSQL.appendChild(btnFromSQL_postgres); - const btnFromSQL_sqlite = mxUtils.button('Insert Sqlite', function() { - parseSql(sqlInputFromSQL.value, 'sqlite'); + const btnFromSQL_sqlite = mxUtils.button("Insert Sqlite", function() { + parseSql(sqlInputFromSQL.value, "sqlite"); }); - btnFromSQL_sqlite.style.marginTop = '8px'; - btnFromSQL_sqlite.style.padding = '4px'; + btnFromSQL_sqlite.style.marginTop = "8px"; + btnFromSQL_sqlite.style.padding = "4px"; divFromSQL.appendChild(btnFromSQL_sqlite); // Adds action - ui.actions.addAction('fromSql', function() { + ui.actions.addAction("fromSql", function() { wndFromSQL.setVisible(!wndFromSQL.isVisible()); if (wndFromSQL.isVisible()) { @@ -633,13 +632,13 @@ Draw.loadPlugin(function(ui) { // end import diagrams from sql text methods // finalize menu buttons - var theMenu = ui.menus.get('insert'); + const theMenu = ui.menus.get("insert"); if(theMenu && theMenu.enabled) { - var oldMenu = theMenu.funct; + const oldMenu = theMenu.funct; theMenu.funct = function(...args) { const [menu, parent] = args; oldMenu.apply(this, args); - ui.menus.addMenuItems(menu, ['fromSql'], parent); + ui.menus.addMenuItems(menu, ["fromSql"], parent); }; } if(theMenuExportAs && theMenuExportAs.enabled) { @@ -648,18 +647,17 @@ Draw.loadPlugin(function(ui) { theMenuExportAs.funct = function(...args) { const [menu, parent] = args; oldMenuExportAs.apply(this, args); - ui.menus.addMenuItems(menu, ['tosql'], parent); + ui.menus.addMenuItems(menu, ["tosql"], parent); }; } else { // vscode file export sql menu - var menu = ui.menus.get('file'); + const menu = ui.menus.get("file"); if(menu && menu.enabled) { var oldMenuExportAs = menu.funct; menu.funct = function(...args) { const [menu, parent] = args; oldMenuExportAs.apply(this, args); - debugger - ui.menus.addMenuItems(menu, ['tosql'], parent); + ui.menus.addMenuItems(menu, ["tosql"], parent); }; } } diff --git a/src/types/drawio-types.d.ts b/src/types/drawio-types.d.ts index 493267f..84942a4 100644 --- a/src/types/drawio-types.d.ts +++ b/src/types/drawio-types.d.ts @@ -20,7 +20,7 @@ declare class mxCell { * @param geometry Optional that specifies the geometry. * @param style Optional formatted string that defines the style. */ - constructor(value?:string, geometry?:any, style?: string); + constructor(value?: string, geometry?: any, style?: string); public vertex: boolean; public connectable: boolean; @@ -42,7 +42,7 @@ declare class mxRectangle { * @param width * @param height */ - constructor(x:number, y:number, width:number, height:number); + constructor(x: number, y: number, width: number, height: number); public width: number; public height: number; @@ -51,7 +51,7 @@ declare class mxRectangle { } declare class mxGeometry extends mxRectangle { - constructor(x:number, y:number, width:number, height:number); + constructor(x: number, y: number, width: number, height: number); } @@ -61,7 +61,7 @@ declare class mxPoint { * @param x * @param y */ - constructor(x:number, y:number); + constructor(x: number, y: number); public x: number; public y: number; @@ -73,7 +73,7 @@ declare class sb { * @param cell * @param value */ - static cloneCell(cell:mxCell, value:string): mxCell; + static cloneCell(cell: mxCell, value: string): mxCell; } declare class mxResources { @@ -169,7 +169,7 @@ declare interface DrawioGraph { setSelectionCells(cells: DrawioCell[]); importCells(cells: mxCell[], x: number, y: number); getPreferredSizeForCell(cell: mxCell): CellSize; - insertEdge(parent: mxCell | null, id:string | null, value:string, source?: mxCell, target?: mxCell, style?:string): DrawioCell; + insertEdge(parent: mxCell | null, id: string | null, value: string, source?: mxCell, target?: mxCell, style?: string): DrawioCell; getModel(): DrawioGraphModel; getFreeInsertPoint(): mxPoint; getSelectionCell(): DrawioCell; @@ -180,7 +180,7 @@ declare interface DrawioGraph { addMouseListener(listener: { mouseMove?: (graph: DrawioGraph, event: mxMouseEvent) => void; - mouseDown?: (graph: DrawioGraph, event: mxMouseEvent) => void + mouseDown?: (graph: DrawioGraph, event: mxMouseEvent) => void; mouseUp?: (graph: DrawioGraph, event: mxMouseEvent) => void; }): void; } From 3ad51110e25121426992d044d0f26157f88e94c7 Mon Sep 17 00:00:00 2001 From: lastlink Date: Sun, 23 Jun 2024 09:09:25 -0400 Subject: [PATCH 06/12] almost there --- README.md | 6 ++++- build/updateVersion.js | 20 +++++++++++++++ lib/package.json | 55 ++++++++++++++++++++++++++++++++++++++++++ package.json | 5 ++-- src/sql.ts | 5 ++-- tsconfig.json | 2 +- 6 files changed, 87 insertions(+), 6 deletions(-) create mode 100644 build/updateVersion.js create mode 100644 lib/package.json diff --git a/README.md b/README.md index 4544b97..1bb008a 100644 --- a/README.md +++ b/README.md @@ -2,4 +2,8 @@ plugins for sql tooling in drawio ## Getting Started -* see https://github.com/ariel-bentu/tam-drawio for install options with vscode \ No newline at end of file +* see https://github.com/ariel-bentu/tam-drawio for install options with vscode + +## Development +* `npm install` +* `npm build:client` to update `dist/sql.js` \ No newline at end of file diff --git a/build/updateVersion.js b/build/updateVersion.js new file mode 100644 index 0000000..b6a96e6 --- /dev/null +++ b/build/updateVersion.js @@ -0,0 +1,20 @@ +const package = require('../package.json'); +const fs = require("fs"); +const directoryPath = "./dist"; // Change this to your actual directory path + +const files = fs.readdirSync(directoryPath); +console.log("Files in the directory:", files); + +const oldText = ""; +const newText = package.VERSION; + +console.log("Updating to version " + package.version); + +files.forEach((file) => { + const filePath = `${directoryPath}/${file}`; + let content = fs.readFileSync(filePath, "utf8"); + content = content.replace(oldText, package.version); + // content = content.replace(new RegExp(oldText, "g"), newText); + fs.writeFileSync(filePath, content, "utf8"); + console.log(`Replaced text in ${file}`); +}); diff --git a/lib/package.json b/lib/package.json new file mode 100644 index 0000000..67cd29b --- /dev/null +++ b/lib/package.json @@ -0,0 +1,55 @@ +{ + "name": "sqltooling-drawio", + "version": "0.0.1", + "description": "plugins for sql tooling in drawio", + "main": "index.js", + "engines": { + "node": ">=16.x" + }, + "scripts": { + "start": "node index.mjs", + "test": "jest -c ./jest.config.ts --forceExit --verbose -i --no-cache --detectOpenHandles", + "test:coverage": "jest --forceExit --coverage --verbose --detectOpenHandles", + "test:watch": "jest --watchAll --detectOpenHandles", + "lint": "tsc --noEmit && eslint \"{src,client}/**/*.{js,ts}\"", + "lint:fix": "tsc --noEmit && eslint \"{src,client}/**/*.{js,ts}\" --fix", + "create": "npm run build && npm run cli:test", + "build": "tsc && npm run build:types", + "build:types": "tsc -p tsconfig-types.json", + "build:help": "tsc --help", + "build:client": "npm run mkdir:dist && browserify src/index.ts -p [ tsify --noImplicitAny ] > dist/littleMermaid.js", + "mkdir:dist": "node ./build/createDistFolder.js" + }, + "repository": { + "type": "git", + "url": "git+https://github.com/lastlink/sqltooling-drawio.git" + }, + "keywords": [ + "uml-diagram", + "drawio", + "sqltools" + ], + "author": "lastlink", + "license": "MIT", + "bugs": { + "url": "https://github.com/lastlink/sqltooling-drawio/issues" + }, + "homepage": "https://github.com/lastlink/sqltooling-drawio#readme", + "dependencies": { + "@funktechno/little-mermaid-2-the-sql": "^0.1.1", + "@funktechno/sqlsimpleparser": "^0.1.0" + }, + "devDependencies": { + "@types/jest": "^29.0.3", + "jest": "^29.0.3", + "jest-junit": "^16.0.0", + "jest-serial-runner": "^1.1.0", + "ts-jest": "^29.0.1", + "ts-node": "^10.9.1", + "typescript": "^5.0.2" + }, + "directories": { + "lib": "lib", + "test": "tests" + } +} diff --git a/package.json b/package.json index c623db7..ad57de6 100644 --- a/package.json +++ b/package.json @@ -17,8 +17,9 @@ "build": "tsc && npm run build:types", "build:types": "tsc -p tsconfig-types.json", "build:help": "tsc --help", - "build:client": "npm run mkdir:dist && browserify src/sql.ts -p [ tsify --noImplicitAny ] > dist/sql.js", - "mkdir:dist": "node ./build/createDistFolder.js" + "build:client": "npm run mkdir:dist && browserify src/sql.ts -p [ tsify --noImplicitAny ] > dist/sql.js && npm run updateVersion:dist", + "mkdir:dist": "node ./build/createDistFolder.js", + "updateVersion:dist": "node ./build/updateVersion.js" }, "repository": { "type": "git", diff --git a/src/sql.ts b/src/sql.ts index 1f69623..088ec85 100644 --- a/src/sql.ts +++ b/src/sql.ts @@ -5,7 +5,8 @@ import { SqlSimpleParser } from "@funktechno/sqlsimpleparser"; import { ForeignKeyModel, PrimaryKeyModel, PropertyModel, TableModel } from "@funktechno/sqlsimpleparser/lib/types"; /** - * Parse SQL CREATE TABLE. Simple initial version for community to improve. + * SQL Tools Plugin for importing diagrams from SQL DDL and exporting to SQL. + * Version: */ Draw.loadPlugin(function(ui) { @@ -633,7 +634,7 @@ Draw.loadPlugin(function(ui) { // finalize menu buttons const theMenu = ui.menus.get("insert"); - if(theMenu && theMenu.enabled) { + if(theMenu) { const oldMenu = theMenu.funct; theMenu.funct = function(...args) { const [menu, parent] = args; diff --git a/tsconfig.json b/tsconfig.json index 90ac91a..6989c39 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -112,7 +112,7 @@ "deps", "node_modules", "**/tests/*", - "src/*.spec.ts", + "src/*.spec.ts" // "src/mermaid" ] } From 1100cda7d371872a3ba128d0c8aafe941a64c6d9 Mon Sep 17 00:00:00 2001 From: lastlink Date: Sun, 23 Jun 2024 09:10:05 -0400 Subject: [PATCH 07/12] build test --- .github/workflows/tests.yaml | 1 + 1 file changed, 1 insertion(+) diff --git a/.github/workflows/tests.yaml b/.github/workflows/tests.yaml index 1870f3c..ccfc885 100644 --- a/.github/workflows/tests.yaml +++ b/.github/workflows/tests.yaml @@ -19,3 +19,4 @@ jobs: node-version: current - run: npm ci - run: npm test + - run: npm run build:client From b3b351b4349f0bb8e07d7a0f5c33412158c88d7c Mon Sep 17 00:00:00 2001 From: lastlink Date: Sun, 23 Jun 2024 09:11:24 -0400 Subject: [PATCH 08/12] delete extra --- lib/package.json | 55 ------------------------------------------------ 1 file changed, 55 deletions(-) delete mode 100644 lib/package.json diff --git a/lib/package.json b/lib/package.json deleted file mode 100644 index 67cd29b..0000000 --- a/lib/package.json +++ /dev/null @@ -1,55 +0,0 @@ -{ - "name": "sqltooling-drawio", - "version": "0.0.1", - "description": "plugins for sql tooling in drawio", - "main": "index.js", - "engines": { - "node": ">=16.x" - }, - "scripts": { - "start": "node index.mjs", - "test": "jest -c ./jest.config.ts --forceExit --verbose -i --no-cache --detectOpenHandles", - "test:coverage": "jest --forceExit --coverage --verbose --detectOpenHandles", - "test:watch": "jest --watchAll --detectOpenHandles", - "lint": "tsc --noEmit && eslint \"{src,client}/**/*.{js,ts}\"", - "lint:fix": "tsc --noEmit && eslint \"{src,client}/**/*.{js,ts}\" --fix", - "create": "npm run build && npm run cli:test", - "build": "tsc && npm run build:types", - "build:types": "tsc -p tsconfig-types.json", - "build:help": "tsc --help", - "build:client": "npm run mkdir:dist && browserify src/index.ts -p [ tsify --noImplicitAny ] > dist/littleMermaid.js", - "mkdir:dist": "node ./build/createDistFolder.js" - }, - "repository": { - "type": "git", - "url": "git+https://github.com/lastlink/sqltooling-drawio.git" - }, - "keywords": [ - "uml-diagram", - "drawio", - "sqltools" - ], - "author": "lastlink", - "license": "MIT", - "bugs": { - "url": "https://github.com/lastlink/sqltooling-drawio/issues" - }, - "homepage": "https://github.com/lastlink/sqltooling-drawio#readme", - "dependencies": { - "@funktechno/little-mermaid-2-the-sql": "^0.1.1", - "@funktechno/sqlsimpleparser": "^0.1.0" - }, - "devDependencies": { - "@types/jest": "^29.0.3", - "jest": "^29.0.3", - "jest-junit": "^16.0.0", - "jest-serial-runner": "^1.1.0", - "ts-jest": "^29.0.1", - "ts-node": "^10.9.1", - "typescript": "^5.0.2" - }, - "directories": { - "lib": "lib", - "test": "tests" - } -} From 638d13cdba7dc848be271db703b8716545a066e7 Mon Sep 17 00:00:00 2001 From: lastlink Date: Sun, 23 Jun 2024 09:11:49 -0400 Subject: [PATCH 09/12] ignore lib --- .gitignore | 1 + 1 file changed, 1 insertion(+) diff --git a/.gitignore b/.gitignore index ffbcb58..1049857 100644 --- a/.gitignore +++ b/.gitignore @@ -128,3 +128,4 @@ out .yarn/install-state.gz .pnp.* junit.xml +lib \ No newline at end of file From cb9fa3e40e974a19e76c7114a05050d47abbead7 Mon Sep 17 00:00:00 2001 From: lastlink Date: Sun, 23 Jun 2024 11:02:42 -0400 Subject: [PATCH 10/12] rm debugger --- dist/sql.js | 1 - 1 file changed, 1 deletion(-) diff --git a/dist/sql.js b/dist/sql.js index 9f8aebb..e64523c 100644 --- a/dist/sql.js +++ b/dist/sql.js @@ -1452,7 +1452,6 @@ Draw.loadPlugin(function(ui) { var oldMenuExportAs = menu.funct; menu.funct = function(menu, parent) { oldMenuExportAs.apply(this, arguments); - debugger ui.menus.addMenuItems(menu, ['tosql'], parent); }; } From 8015f79917898323837fb899dc100282e33e4717 Mon Sep 17 00:00:00 2001 From: lastlink Date: Sun, 23 Jun 2024 11:58:19 -0400 Subject: [PATCH 11/12] working plugin --- dist/sql.js | 2419 ++++++++++++++++++++++++++------------------------- src/sql.ts | 24 +- 2 files changed, 1229 insertions(+), 1214 deletions(-) diff --git a/dist/sql.js b/dist/sql.js index e64523c..0dc0f4f 100644 --- a/dist/sql.js +++ b/dist/sql.js @@ -1,188 +1,943 @@ +(function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i 0) { + attributeName = testFullNameMatches[0]; + attributeComment = attributeComment + .replace(`'${attributeName}'`, "") + .trim(); + } + } + if (attributeComment) + attributeComment = " " + attributeComment; + // check if contains full column name + statement += `\t${this.dbTypeEnds(attributeName)} ${columnType}${attributeComment}`; + if (attribute.attributeKeyType && + attribute.attributeKeyType == "PK") { + primaryKeys.push(attribute.attributeName); } } + else { + if (attribute.attributeKeyType && + attribute.attributeKeyType == "PK") { + primaryKeys.push(attribute.attributeName); + } + statement += `\t${this.dbTypeEnds(attribute.attributeName)} ${columnType}`; + } + // statement += i != entity.attributes.length - 1 ? ",\n" : "\n"; } - return statementGeneration.join(""); } - /** - * convert labels with start and end strings per database type - * @param label - * @returns - */ - dbTypeEnds(label) { - let char1 = '"'; - let char2 = '"'; - if (this.dbType == "mysql") { - char1 = "`"; - char2 = "`"; + if (primaryKeys.length > 0) { + statement += ",\n\tPRIMARY KEY("; + for (let i = 0; i < primaryKeys.length; i++) { + statement += (i == 0 ? "" : ",") + this.dbTypeEnds(primaryKeys[i]); } - else if (this.dbType == "sqlserver") { - char1 = "["; - char2 = "]"; + statement += ")"; + } + // foreign keys + const entityFKeys = this.relationships.filter((relation) => relation.entityB == entityKey); + if (entityFKeys.length > 0) { + for (let i = 0; i < entityFKeys.length; i++) { + const fk = entityFKeys[i]; + const fkRelTxt = fk.roleA; + // must match format "[..] to [..]" + const keySplit = "] to ["; + if (fkRelTxt.indexOf(keySplit) != -1 && + fkRelTxt[0] == "[" && + fkRelTxt[fkRelTxt.length - 1] == "]") { + const keys = fkRelTxt.substring(1, fkRelTxt.length - 1).split(keySplit); + // remove quotes + let fkCol = keys[1].replace(/[\'\"]/gim, ""); + if (fkCol.indexOf(".") != -1) { + fkCol = fkCol.split(".")[1]; + } + fkCol = fkCol.trim(); + let pkCol = keys[0].replace(/[\'\"]/gim, ""); + if (pkCol.indexOf(".") != -1) { + pkCol = pkCol.split(".")[1]; + } + pkCol = pkCol.trim(); + // FOREIGN KEY (`Artist Id`) REFERENCES `Artist`(`ArtistId`) + statement += `,\n\tFOREIGN KEY (${this.dbTypeEnds(fkCol)}) REFERENCES ${this.dbTypeEnds(fk.entityA)}(${this.dbTypeEnds(pkCol)})`; + } } - return `${char1}${label}${char2}`; } + if (attributesAdded != 0) { + statement += "\n"; + } + statement += ");\n\n"; + return statement; + } + /* database create table examples: + sqlite example: + CREATE TABLE [test_table] ( + "id" INTEGER NOT NULL, + "Field 2_2" TEXT, + "Artist Id" INTEGER, + PRIMARY KEY("id"), + FOREIGN KEY("Artist Id") REFERENCES "Artist"("ArtistId") ON DELETE NO ACTION ON UPDATE NO ACTION + ) + + mysql example: + CREATE TABLE `test_table` ( + `id` INTEGER NOT NULL, + `Field 2_2` TEXT, + `Artist Id` INTEGER, + PRIMARY KEY (`id`), + FOREIGN KEY (`Artist Id`) REFERENCES `Artist`(`ArtistId`) + ); + + postgres example: + CREATE TABLE "test_table" ( + "id" INTEGER NOT NULL, + "Field 2_2" TEXT, + "Artist Id" INTEGER, + PRIMARY KEY ("id"), + FOREIGN KEY ("Artist Id"") REFERENCES "Artist"("ArtistId") + ); + + sql server example: + CREATE TABLE [test_table] ( + [id] INTEGER NOT NULL, + [Field 2_2] TEXT, + [Artist Id] INTEGER, + PRIMARY KEY ([id]), + FOREIGN KEY ([Artist Id]) REFERENCES [Artist]([ArtistId]) + ); + */ + createColumn(tableName, columnName, dataType) { + return `ALTER TABLE ${tableName} ADD ${columnName} ${dataType}`; + } +} +exports.DbParser = DbParser; + +},{}],2:[function(require,module,exports){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +exports.CONSTRAINT_Foreign_Key = exports.CONSTRAINT_Primary_Key = exports.Foreign_Key = exports.Primary_Key = exports.CONSTRAINT = exports.AlterTable = exports.CreateTable = void 0; +exports.CreateTable = "create table"; +exports.AlterTable = "alter table "; +exports.CONSTRAINT = "constraint"; +exports.Primary_Key = "primary key"; +exports.Foreign_Key = "foreign key"; +exports.CONSTRAINT_Primary_Key = "constraint primary key"; +exports.CONSTRAINT_Foreign_Key = "constraint foreign key"; + +},{}],3:[function(require,module,exports){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +exports.SqlSimpleParser = void 0; +const contants_1 = require("./contants"); +/* +An attempt to fix sql parser for importing entity diagrams into draw io see +https://github.com/jgraph/drawio/issues/1178 +https://drawio-app.com/entity-relationship-diagrams-with-draw-io/ + +*/ +/** + * Main Parser class + */ +class SqlSimpleParser { + /** + * Parser constructor. + * Default dialect is 'sqlite'. + * Options: "mysql" | "sqlite" | "postgres" | "sqlserver" . + * + * @param dialect SQL dialect ('sqlite'). + */ + constructor(dialect = "sqlite") { + this.tableList = []; + /** + * Parsed statements. + */ + this.statements = []; /** - * generate create table statement - * also includes primary keys and foreign keys - * @param entityKey - * @param entity - * @returns + * Remains of string feed, after last parsed statement. */ - createTable(entityKey, entity) { - let statement = `CREATE TABLE ${this.dbTypeEnds(entityKey)} (`; - let primaryKeys = []; - let attributesAdded = 0; - for (let i = 0; i < entity.attributes.length; i++) { - const attribute = entity.attributes[i]; - if (attribute.attributeType && attribute.attributeName) { - statement += attributesAdded == 0 ? "\n" : ",\n"; - attributesAdded++; - // need to add parenthesis or commas - // BIGGEST difference from original is that column types don't have to be fixed - let columnType = attribute.attributeType.trim(); - if (attribute.attributeComment) { - let attributeName = attribute.attributeName; - let attributeComment = attribute.attributeComment.trim(); - if (attribute.attributeComment.indexOf("'") != -1) { - // extract - const testFullNameMatches = attribute.attributeComment.match(/(?<=((?<=[\s,.:;"']|^)["']))(?:(?=(\\?))\2.)*?(?=\1)/gmu); - if (testFullNameMatches && testFullNameMatches.length > 0) { - attributeName = testFullNameMatches[0]; - attributeComment = attributeComment - .replace(`'${attributeName}'`, "") + this.remains = ""; + /** + * Whether preparser is currently escaped. + */ + this.escaped = false; + /** + * Current quote char of preparser. + */ + this.quoted = ""; + this.exportedTables = 0; + this.SQLServer = "sqlserver"; + this.MODE_SQLSERVER = false; + this.foreignKeyList = []; + this.primaryKeyList = []; + this.dialect = dialect; + this.MODE_SQLSERVER = + this.dialect !== undefined && + this.dialect !== null && + this.dialect == this.SQLServer; + } + /** + * Feed chunk of string into parser. + * + * @param chunk Chunk of string to be parsed. + */ + feed(chunk) { + // + const removedComments = chunk + // remove database comments, multiline, --, and // + .replace(/\/\*[\s\S]*?\*\/|\/\/|--.*/g, "") + .replace(/IF NOT EXISTS/gi, "") + .trim(); + const cleanedLines = removedComments + .split("\n") + // remove empty lines + .filter((n) => n) + // remove multiple spaces + .map((n) => n.replace(/\s+/g, " ").trim()); + // combine lines that are in parenthesis + const lines = []; + let insertSameLine = false; + cleanedLines.forEach((n) => { + if (lines.length > 0) { + if ((n[0] == "(" && + lines[lines.length - 1].toLocaleLowerCase().indexOf(contants_1.CreateTable) == + -1) || + insertSameLine) { + if (lines.length > 0) { + insertSameLine = true; + lines[lines.length - 1] += ` ${n}`; + if (n[0] == ")") + insertSameLine = false; + } + } + else if (lines[lines.length - 1].match(/CONSTRAINT/gi) && + (n.match(/FOREIGN KEY/gi) && !n.match(/CONSTRAINT/gi))) { + lines[lines.length - 1] += ` ${n}`; + } + // add to previous line if current has references and previous has foreign key + else if (lines[lines.length - 1].match(/FOREIGN KEY/gi) && + (n.match(/REFERENCES/gi) && !n.match(/FOREIGN KEY/gi))) { + lines[lines.length - 1] += ` ${n}`; + } + else if (n.substring(0, 2).toUpperCase() == "ON") { + lines[lines.length - 1] += ` ${n}`; + } + else { + lines.push(n); + } + } + else { + lines.push(n); + } + }); + // dx = 0, + // tableCell = null, + // cells = [], + // exportedTables = 0, + // tableList = [], + // foreignKeyList = [], + // rowCell = null; + let currentTableModel = null; + //Parse SQL to objects + for (let i = 0; i < lines.length; i++) { + // rowCell = null; + const tmp = lines[i].trim(); + const propertyRow = tmp.toLowerCase().trim(); + if (propertyRow[0] == ")") { + // close table + if (currentTableModel) { + this.tableList.push(currentTableModel); + currentTableModel = null; + } + continue; + } + //Parse Table + if (propertyRow.indexOf(contants_1.CreateTable) != -1) { + //Parse row + let name = tmp + .replace(this.stringToRegex(`/${contants_1.CreateTable}/gi`), "") + .trim(); + //Parse Table Name + name = this.ParseTableName(name); + if (currentTableModel !== null) { + //Add table to the list + this.tableList.push(currentTableModel); + } + //Create Table + currentTableModel = this.CreateTable(name); + } + // Parse Properties + else if (tmp !== "(" && + currentTableModel != null && + propertyRow.indexOf(contants_1.AlterTable) == -1) { + //Parse the row + let name = tmp.substring(0, tmp.charAt(tmp.length - 1) === "," ? tmp.length - 1 : tmp.length); + //Attempt to get the Key Type + let propertyType = name.toLowerCase(); + // .substring(0, AlterTable.length).toLowerCase(); + //Add special constraints + if (this.MODE_SQLSERVER) { + if (propertyType.indexOf(contants_1.CONSTRAINT) !== -1 && + propertyType.indexOf(contants_1.Primary_Key) !== -1) { + propertyType = contants_1.CONSTRAINT_Primary_Key; + } + if (propertyType.indexOf(contants_1.CONSTRAINT) !== -1 && + propertyType.indexOf(contants_1.Foreign_Key) !== -1) { + propertyType = contants_1.CONSTRAINT_Foreign_Key; + } + } + //Verify if this is a property that doesn't have a relationship (One minute of silence for the property) + // TODO: make primary key check a regex match/ contains w/ () space or not + const normalProperty = !propertyType.match(/PRIMARY KEY\s?\(/gi) && + propertyType.indexOf(contants_1.Foreign_Key) == -1 && + propertyType.indexOf(contants_1.CONSTRAINT_Primary_Key) == -1 && + propertyType.indexOf(contants_1.CONSTRAINT_Foreign_Key) == -1; + const nameSkipCheck = name.toUpperCase().trim(); + //Parse properties that don't have relationships + if (normalProperty) { + if (name === "" || name === "" || name === ");") { + continue; + } + let ExtendedProperties = null; + if (this.MODE_SQLSERVER) { + if (nameSkipCheck.indexOf(" ASC") !== -1 || + nameSkipCheck.indexOf(" DESC") !== -1 || + nameSkipCheck.indexOf("EXEC ") !== -1 || + nameSkipCheck.indexOf("WITH ") !== -1 || + nameSkipCheck.indexOf(" ON") !== -1 || + nameSkipCheck.indexOf("ALTER ") !== -1 || + // comments already removed + nameSkipCheck.indexOf("/*") !== -1 || + nameSkipCheck.indexOf(" CONSTRAINT") !== -1 || + nameSkipCheck.indexOf("SET ") !== -1 || + nameSkipCheck.indexOf(" NONCLUSTERED") !== -1 || + // no spaces desired + nameSkipCheck.indexOf("GO") !== -1 || + nameSkipCheck.indexOf("REFERENCES ") !== -1) { + continue; + } + //Get delimiter of column name + //TODO: check for space? or end quantifier + const firstSpaceIndex = name[0] == "[" && name.indexOf("]" + " ") !== -1 + ? name.indexOf("]" + " ") + : name.indexOf(" "); + ExtendedProperties = name.substring(firstSpaceIndex + 1).trim(); + //Get full name + name = name.substring(0, firstSpaceIndex); + name = this.RemoveNameQuantifiers(name); + } + else { + const columnQuantifiers = this.GetColumnQuantifiers(); + //Get delimiter of column name + const firstSpaceIndex = name[0] == columnQuantifiers.Start && + name.indexOf(columnQuantifiers.End + " ") !== -1 + ? name.indexOf(columnQuantifiers.End + " ") + : name.indexOf(" "); + ExtendedProperties = name.substring(firstSpaceIndex + 1).trim(); + //Get full name + name = name.substring(0, firstSpaceIndex); + name = this.RemoveNameQuantifiers(name); + } + //Create Property + const propertyModel = this.CreateProperty(name, currentTableModel.Name, null, false, ExtendedProperties); + //Add Property to table + currentTableModel.Properties.push(propertyModel); + if (ExtendedProperties.toLocaleLowerCase().indexOf(contants_1.Primary_Key) > -1) { + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(name, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + } + } + else { + //Parse Primary Key + if (propertyType.indexOf(contants_1.Primary_Key) != -1 || + propertyType.indexOf(contants_1.CONSTRAINT_Primary_Key) != -1) { + if (!this.MODE_SQLSERVER) { + const primaryKey = name + .replace(/PRIMARY KEY\s?\(/gi, "") + .replace(")", ""); + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(primaryKey, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + } + else { + // let start = i + 2; + // let end = 0; + if (propertyRow.indexOf(contants_1.Primary_Key) !== -1 && + nameSkipCheck.indexOf("CLUSTERED") === -1) { + const primaryKey = name + .replace(/PRIMARY KEY\s?\(/gi, "") + .replace(")", ""); + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(primaryKey, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + } + else { + const startIndex = name.toLocaleLowerCase().indexOf("("); + const endIndex = name.indexOf(")") + 1; + const primaryKey = name + .substring(startIndex, endIndex) + .replace("(", "") + .replace(")", "") + .replace(/ASC/gi, "") .trim(); + const columnQuantifiers = this.GetColumnQuantifiers(); + //Get delimiter of column name + const firstSpaceIndex = primaryKey[0] == columnQuantifiers.Start && + primaryKey.indexOf(columnQuantifiers.End + " ") !== -1 + ? primaryKey.indexOf(columnQuantifiers.End + " ") + : primaryKey.indexOf(" "); + const primaryKeyRow = firstSpaceIndex == -1 + ? primaryKey + : primaryKey.substring(firstSpaceIndex + 1).trim(); + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(primaryKeyRow, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + /* + while (end === 0) { + let primaryKeyRow = lines[start].trim(); + + if (primaryKeyRow.indexOf(")") !== -1) { + end = 1; + break; + } + + start++; + + primaryKeyRow = primaryKeyRow.replace(/ASC/gi, ""); + + //Parse name + primaryKeyRow = this.ParseSQLServerName(primaryKeyRow, true); + + //Create Primary Key + let primaryKeyModel = this.CreatePrimaryKey( + primaryKeyRow, + currentTableModel.Name + ); + + //Add Primary Key to List + this.primaryKeyList.push(primaryKeyModel); + } + */ } } - if (attributeComment) - attributeComment = " " + attributeComment; - // check if contains full column name - statement += `\t${this.dbTypeEnds(attributeName)} ${columnType}${attributeComment}`; - if (attribute.attributeKeyType && - attribute.attributeKeyType == "PK") { - primaryKeys.push(attribute.attributeName); - } + } + } + //Parse Foreign Key + if (propertyType.indexOf(contants_1.Foreign_Key) != -1 || + propertyType.indexOf(contants_1.CONSTRAINT_Foreign_Key) != -1) { + if (!this.MODE_SQLSERVER || propertyRow.indexOf(contants_1.AlterTable) == -1) { + this.ParseMySQLForeignKey(name, currentTableModel); } else { - if (attribute.attributeKeyType && - attribute.attributeKeyType == "PK") { - primaryKeys.push(attribute.attributeName); + let completeRow = name; + if (nameSkipCheck.indexOf("REFERENCES") === -1) { + const referencesRow = lines[i + 1].trim(); + completeRow = + "ALTER TABLE [dbo].[" + + currentTableModel.Name + + "] WITH CHECK ADD" + + " " + + name + + " " + + referencesRow; } - statement += `\t${this.dbTypeEnds(attribute.attributeName)} ${columnType}`; + this.ParseSQLServerForeignKey(completeRow, currentTableModel); } } } - if (primaryKeys.length > 0) { - statement += ",\n\tPRIMARY KEY("; - for (let i = 0; i < primaryKeys.length; i++) { - const element = primaryKeys[i]; - statement += (i == 0 ? "" : ",") + this.dbTypeEnds(primaryKeys[i]); + else if (propertyRow.indexOf(contants_1.AlterTable) != -1) { + if (this.MODE_SQLSERVER) { + //Parse the row + const alterTableRow = tmp.substring(0, tmp.charAt(tmp.length - 1) === "," ? tmp.length - 1 : tmp.length); + const referencesRow = lines[i + 1].trim(); + const completeRow = alterTableRow + " " + referencesRow; + this.ParseSQLServerForeignKey(completeRow, currentTableModel); } - statement += ")"; } - // foreign keys - let entityFKeys = this.relationships.filter((relation) => relation.entityB == entityKey); - if (entityFKeys.length > 0) { - for (let i = 0; i < entityFKeys.length; i++) { - const fk = entityFKeys[i]; - const fkRelTxt = fk.roleA; - // must match format "[..] to [..]" - const keySplit = "] to ["; - if (fkRelTxt.indexOf(keySplit) != -1 && - fkRelTxt[0] == "[" && - fkRelTxt[fkRelTxt.length - 1] == "]") { - let keys = fkRelTxt.substring(1, fkRelTxt.length - 1).split(keySplit); - // remove quotes - let fkCol = keys[1].replace(/[\'\"]/gim, ""); - if (fkCol.indexOf(".") != -1) { - fkCol = fkCol.split(".")[1]; - } - fkCol = fkCol.trim(); - let pkCol = keys[0].replace(/[\'\"]/gim, ""); - if (pkCol.indexOf(".") != -1) { - pkCol = pkCol.split(".")[1]; + } + // parse fk and primary keys + if (this.primaryKeyList.length > 0) { + this.primaryKeyList.forEach((pk) => { + // find table index + const pkTableIndex = this.tableList.findIndex((t) => t.Name.toLocaleLowerCase() == + pk.PrimaryKeyTableName.toLocaleLowerCase()); + // find property index + if (pkTableIndex > -1) { + const propertyIndex = this.tableList[pkTableIndex].Properties.findIndex((p) => p.Name.toLocaleLowerCase() == + pk.PrimaryKeyName.toLocaleLowerCase()); + if (propertyIndex > -1) { + this.tableList[pkTableIndex].Properties[propertyIndex].IsPrimaryKey = true; + } + } + }); + } + if (this.foreignKeyList.length > 0) { + this.foreignKeyList.forEach((fk) => { + // find table index + const pkTableIndex = this.tableList.findIndex((t) => t.Name.toLocaleLowerCase() == + fk.ReferencesTableName.toLocaleLowerCase()); + // let fkTableIndex = this.tableList.findIndex( + // (t) => t.Name == fk.PrimaryKeyTableName + // ); + // find property index + if (pkTableIndex > -1) { + const propertyIndex = this.tableList[pkTableIndex].Properties.findIndex((p) => p.Name.toLocaleLowerCase() == + fk.PrimaryKeyName.toLocaleLowerCase()); + if (propertyIndex > -1) { + this.tableList[pkTableIndex].Properties[propertyIndex].ForeignKey.push(fk); + if (!fk.IsDestination) { + this.tableList[pkTableIndex].Properties[propertyIndex].IsForeignKey = true; } - pkCol = pkCol.trim(); - statement += `,\n\tFOREIGN KEY (${this.dbTypeEnds(fkCol)}) REFERENCES ${this.dbTypeEnds(fk.entityA)}(${this.dbTypeEnds(pkCol)})`; } } + // if (fkTableIndex > -1) { + // let propertyIndex = this.tableList[fkTableIndex].Properties.findIndex( + // (p) => p.Name == fk.PrimaryKeyName + // ); + // if (propertyIndex > -1) { + // this.tableList[fkTableIndex].Properties[propertyIndex].ForeignKey.push(fk) + // } + // } + }); + } + return this; + } + stringToRegex(str) { + // Main regex + const mainResult = str.match(/\/(.+)\/.*/); + const optionsResult = str.match(/\/.+\/(.*)/); + if (mainResult && optionsResult) { + const main = mainResult[1]; + // Regex options + const options = optionsResult[1]; + // Compiled regex + return new RegExp(main, options); + } + return new RegExp("//(.+)/.*/", "//.+/(.*)/"); + } + CreatePrimaryKey(primaryKeyName, primaryKeyTableName) { + const primaryKeyNames = this.RemoveNameQuantifiers(primaryKeyName) + .split(",") + .filter((n) => n) + // remove multiple spaces + .map((n) => n.replace(/\s+/g, " ").trim()); + const primaryKeys = []; + primaryKeyNames.forEach(name => { + const primaryKey = { + PrimaryKeyTableName: primaryKeyTableName, + PrimaryKeyName: name, + }; + primaryKeys.push(primaryKey); + }); + return primaryKeys; + } + CreateProperty(name, tableName, foreignKey, isPrimaryKey, columnProps) { + const isForeignKey = foreignKey !== undefined && foreignKey !== null; + const property = { + Name: name, + ColumnProperties: columnProps, + TableName: tableName, + ForeignKey: foreignKey || [], + IsForeignKey: isForeignKey, + IsPrimaryKey: isPrimaryKey, + }; + return property; + } + ParseMySQLForeignKey(name, currentTableModel) { + const referencesIndex = name.toLowerCase().indexOf("references"); + let foreignKeySQL = name.substring(0, referencesIndex); + foreignKeySQL = foreignKeySQL.substring(foreignKeySQL.toUpperCase().indexOf("FOREIGN KEY")); + let referencesSQL = name.substring(referencesIndex, name.length); + //Remove references syntax + referencesSQL = referencesSQL.replace(/REFERENCES /gi, ""); + //Get Table and Property Index + const referencedTableIndex = referencesSQL.indexOf("("); + const referencedPropertyIndex = referencesSQL.indexOf(")"); + //Get Referenced Table + const referencedTableName = referencesSQL.substring(0, referencedTableIndex); + //Get Referenced Key + const referencedPropertyName = referencesSQL.substring(referencedTableIndex + 1, referencedPropertyIndex); + //Get ForeignKey + const foreignKey = foreignKeySQL + .replace(/FOREIGN KEY\s?\(/gi, "") + .replace(")", "") + .replace(" ", ""); + //Create ForeignKey + const foreignKeyOriginModel = this.CreateForeignKey(foreignKey, currentTableModel.Name, referencedPropertyName, referencedTableName, true); + //Add ForeignKey Origin + this.foreignKeyList.push(foreignKeyOriginModel); + //Create ForeignKey + const foreignKeyDestinationModel = this.CreateForeignKey(referencedPropertyName, referencedTableName, foreignKey, currentTableModel.Name, false); + //Add ForeignKey Destination + this.foreignKeyList.push(foreignKeyDestinationModel); + } + ParseSQLServerForeignKey(name, currentTableModel) { + const referencesIndex = name.toLowerCase().indexOf("references"); + let foreignKeySQL = ""; + if (name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`) !== -1) { + foreignKeySQL = name + .substring(name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`), referencesIndex) + .replace(/FOREIGN KEY\(/gi, "") + .replace(")", ""); + } + else { + foreignKeySQL = name + .substring(name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`), referencesIndex) + .replace(/FOREIGN KEY\s?\(/gi, "") + .replace(")", ""); + } + let referencesSQL = name.substring(referencesIndex, name.length); + const nameSkipCheck = name.toUpperCase().trim(); + let alterTableName = name + .substring(0, nameSkipCheck.indexOf("WITH")) + .replace(/ALTER TABLE /gi, ""); + if (referencesIndex !== -1 && + alterTableName !== "" && + foreignKeySQL !== "" && + referencesSQL !== "") { + //Remove references syntax + referencesSQL = referencesSQL.replace(/REFERENCES /gi, ""); + //Get Table and Property Index + const referencedTableIndex = referencesSQL.indexOf("("); + const referencedPropertyIndex = referencesSQL.indexOf(")"); + //Get Referenced Table + let referencedTableName = referencesSQL.substring(0, referencedTableIndex); + //Parse Name + referencedTableName = this.ParseSQLServerName(referencedTableName); + //Get Referenced Key + let referencedPropertyName = referencesSQL.substring(referencedTableIndex + 1, referencedPropertyIndex); + //Parse Name + referencedPropertyName = this.ParseSQLServerName(referencedPropertyName); + //Get ForeignKey + let foreignKey = foreignKeySQL + .replace(/FOREIGN KEY\s?\(/gi, "") + .replace(")", ""); + //Parse Name + foreignKey = this.ParseSQLServerName(foreignKey); + //Parse Name + alterTableName = this.ParseSQLServerName(alterTableName); + //Create ForeignKey + const foreignKeyOriginModel = this.CreateForeignKey(foreignKey, alterTableName, referencedPropertyName, referencedTableName, true); + //Add ForeignKey Origin + this.foreignKeyList.push(foreignKeyOriginModel); + //Create ForeignKey + const foreignKeyDestinationModel = this.CreateForeignKey(referencedPropertyName, referencedTableName, foreignKey, alterTableName, false); + //Add ForeignKey Destination + this.foreignKeyList.push(foreignKeyDestinationModel); + } + } + CreateForeignKey(primaryKeyName, primaryKeyTableName, referencesPropertyName, referencesTableName, isDestination) { + const foreignKey = { + PrimaryKeyTableName: this.RemoveNameQuantifiers(primaryKeyTableName), + PrimaryKeyName: this.RemoveNameQuantifiers(primaryKeyName), + ReferencesPropertyName: this.RemoveNameQuantifiers(referencesPropertyName), + ReferencesTableName: this.RemoveNameQuantifiers(referencesTableName), + IsDestination: isDestination !== undefined && isDestination !== null + ? isDestination + : false, + }; + return foreignKey; + } + RemoveNameQuantifiers(name) { + return name.replace(/\[|\]|\(|\"|\'|\`/g, "").trim(); + } + ParseTableName(name) { + if (name.charAt(name.length - 1) === "(") { + name = this.RemoveNameQuantifiers(name); + // if (!this.MODE_SQLSERVER) { + // name = name.substring(0, name.lastIndexOf(" ")); + // } else { + // name = this.ParseSQLServerName(name); + // } + } + return name; + } + ParseSQLServerName(name, property) { + name = name.replace("[dbo].[", ""); + name = name.replace("](", ""); + name = name.replace("].[", "."); + name = name.replace("[", ""); + if (property == undefined || property == null) { + name = name.replace(" [", ""); + name = name.replace("] ", ""); + } + else { + if (name.indexOf("]") !== -1) { + name = name.substring(0, name.indexOf("]")); } - if (attributesAdded != 0) { - statement += "\n"; - } - statement += `);\n\n`; - return statement; } + if (name.lastIndexOf("]") === name.length - 1) { + name = name.substring(0, name.length - 1); + } + if (name.lastIndexOf(")") === name.length - 1) { + name = name.substring(0, name.length - 1); + } + if (name.lastIndexOf("(") === name.length - 1) { + name = name.substring(0, name.length - 1); + } + name = name.replace(" ", ""); + return name; } - + CreateTable(name) { + const table = { + Name: name, + Properties: [], + }; + //Count exported tables + this.exportedTables++; + return table; + } + /** + * Checks whether character is a quotation character. + * + * @param char Character to be evaluated. + */ + static isQuoteChar(char) { + return char === "\"" || char === "'" || char === "`"; + } + /** + * convert labels with start and end strings per database type + * @param label + * @returns + */ + dbTypeEnds(label) { + let char1 = "\""; + let char2 = "\""; + if (this.dialect == "mysql") { + char1 = "`"; + char2 = "`"; + } + else if (this.dialect == "sqlserver") { + char1 = "["; + char2 = "]"; + } + return `${char1}${label}${char2}`; + } + WithEnds() { + this.tableList = this.tableList.map((table) => { + table.Name = this.dbTypeEnds(table.Name); + table.Properties = table.Properties.map((property) => { + property.Name = this.dbTypeEnds(property.Name); + property.TableName = this.dbTypeEnds(property.TableName); + return property; + }); + return table; + }); + this.primaryKeyList = this.primaryKeyList.map((primaryKey) => { + primaryKey.PrimaryKeyName = this.dbTypeEnds(primaryKey.PrimaryKeyName); + primaryKey.PrimaryKeyTableName = this.dbTypeEnds(primaryKey.PrimaryKeyTableName); + return primaryKey; + }); + this.foreignKeyList = this.foreignKeyList.map((foreignKey) => { + foreignKey.PrimaryKeyName = this.dbTypeEnds(foreignKey.PrimaryKeyName); + foreignKey.ReferencesPropertyName = this.dbTypeEnds(foreignKey.ReferencesPropertyName); + foreignKey.PrimaryKeyTableName = this.dbTypeEnds(foreignKey.PrimaryKeyTableName); + foreignKey.ReferencesTableName = this.dbTypeEnds(foreignKey.ReferencesTableName); + return foreignKey; + }); + return this; + } + WithoutEnds() { + this.tableList.map((table) => { + table.Name = this.RemoveNameQuantifiers(table.Name); + table.Properties = table.Properties.map((property) => { + property.Name = this.RemoveNameQuantifiers(property.Name); + property.TableName = this.RemoveNameQuantifiers(property.TableName); + return property; + }); + return table; + }); + this.primaryKeyList = this.primaryKeyList.map((primaryKey) => { + primaryKey.PrimaryKeyName = this.RemoveNameQuantifiers(primaryKey.PrimaryKeyName); + primaryKey.PrimaryKeyTableName = this.RemoveNameQuantifiers(primaryKey.PrimaryKeyTableName); + return primaryKey; + }); + this.foreignKeyList = this.foreignKeyList.map((foreignKey) => { + foreignKey.PrimaryKeyName = this.RemoveNameQuantifiers(foreignKey.PrimaryKeyName); + foreignKey.ReferencesPropertyName = this.RemoveNameQuantifiers(foreignKey.ReferencesPropertyName); + foreignKey.PrimaryKeyTableName = this.RemoveNameQuantifiers(foreignKey.PrimaryKeyTableName); + foreignKey.ReferencesTableName = this.RemoveNameQuantifiers(foreignKey.ReferencesTableName); + return foreignKey; + }); + return this; + } + /** + * return text quantifiers for dialect + * @returns json + */ + GetColumnQuantifiers() { + const chars = { + Start: "\"", + End: "\"", + }; + if (this.dialect == "mysql") { + chars.Start = "`"; + chars.End = "`"; + } + else if (this.dialect == "sqlserver") { + chars.Start = "["; + chars.End = "]"; + } + return chars; + } + ToTableList() { + return this.tableList; + } + ToPrimaryKeyList() { + return this.primaryKeyList; + } + ToForeignKeyList() { + return this.foreignKeyList; + } + ResetModel() { + this.tableList = []; + this.primaryKeyList = []; + this.foreignKeyList = []; + return this; + } + /** + * return full sql model + * @returns + */ + ToModel() { + return { + TableList: this.tableList, + Dialect: this.dialect, + ForeignKeyList: this.foreignKeyList, + PrimaryKeyList: this.primaryKeyList, + }; + } +} +exports.SqlSimpleParser = SqlSimpleParser; + +},{"./contants":2}],4:[function(require,module,exports){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +const generate_sql_ddl_1 = require("@funktechno/little-mermaid-2-the-sql/lib/src/generate-sql-ddl"); +const sqlsimpleparser_1 = require("@funktechno/sqlsimpleparser"); +/** + * SQL Tools Plugin for importing diagrams from SQL DDL and exporting to SQL. + * Version: 0.0.1 + */ +Draw.loadPlugin(function (ui) { + // export sql methods + /** + * Mermaid Models TO SQL parser + * src https://github.com/Software-Developers-IRL/Little-Mermaid-2-The-SQL/blob/main/src/generate-sql-ddl.ts + */ + // DbParser //Create Base div - const divGenSQL = document.createElement('div'); - divGenSQL.style.userSelect = 'none'; - divGenSQL.style.overflow = 'hidden'; - divGenSQL.style.padding = '10px'; - divGenSQL.style.height = '100%'; - - const sqlInputGenSQL = document.createElement('textarea'); - sqlInputGenSQL.style.height = '200px'; - sqlInputGenSQL.style.width = '100%'; - sqlInputGenSQL.value = '-- click a database type button' + const divGenSQL = document.createElement("div"); + divGenSQL.style.userSelect = "none"; + divGenSQL.style.overflow = "hidden"; + divGenSQL.style.padding = "10px"; + divGenSQL.style.height = "100%"; + const sqlInputGenSQL = document.createElement("textarea"); + sqlInputGenSQL.style.height = "200px"; + sqlInputGenSQL.style.width = "100%"; + sqlInputGenSQL.value = "-- click a database type button"; mxUtils.br(divGenSQL); divGenSQL.appendChild(sqlInputGenSQL); - var theMenuExportAs = ui.menus.get('exportAs'); - let buttonLabel = 'tosql=To SQL' + const theMenuExportAs = ui.menus.get("exportAs"); + let buttonLabel = "tosql=To SQL"; // vscode extension support - if(!(theMenuExportAs && theMenuExportAs.enabled)) { - buttonLabel = 'tosql=Export As SQL' + if (!(theMenuExportAs && theMenuExportAs.enabled)) { + buttonLabel = "tosql=Export As SQL"; } // Extends Extras menu mxResources.parse(buttonLabel); - - const wndGenSQL = new mxWindow(mxResources.get('tosql'), divGenSQL, document.body.offsetWidth - 480, 140, - 320, 320, true, true); + const wndGenSQL = new mxWindow(mxResources.get("tosql"), divGenSQL, document.body.offsetWidth - 480, 140, 320, 320, true, true); wndGenSQL.destroyOnClose = false; wndGenSQL.setMaximizable(false); wndGenSQL.setResizable(false); wndGenSQL.setClosable(true); - /** * return text quantifiers for dialect * @returns json */ function GetColumnQuantifiers(type) { - let chars = { - Start: '"', - End: '"', + const chars = { + Start: "\"", + End: "\"", }; if (type == "mysql") { chars.Start = "`"; @@ -196,1167 +951,434 @@ Draw.loadPlugin(function(ui) { } /** * sometimes rows have spans or styles, an attempt to remove them - * @param {*} label - * @returns + * @param {*} label + * @returns */ - function removeHtml(label){ - var div = document.createElement("div"); - divGenSQL.innerHTML = label; - var text = div.textContent || div.innerText || ""; + function removeHtml(label) { + const tempDiv = document.createElement("div"); + tempDiv.innerHTML = label; + const text = tempDiv.textContent || tempDiv.innerText || ""; + tempDiv.remove(); return text; } /** * extract row column attributes - * @param {*} label - * @param {*} columnQuantifiers - * @returns + * @param {*} label + * @param {*} columnQuantifiers + * @returns */ - function getDbLabel(label, columnQuantifiers){ - label = removeHtml(label) + function getDbLabel(label, columnQuantifiers) { + let result = removeHtml(label); // fix duplicate spaces and different space chars - label = label - .replace(/\s+/g, " ") - let firstSpaceIndex = label[0] == columnQuantifiers.Start && - label.indexOf(columnQuantifiers.End + " ") !== -1 - ? label.indexOf(columnQuantifiers.End + " ") - : label.indexOf(" "); - let attributeType = label.substring(firstSpaceIndex + 1).trim(); - let attributeName = RemoveNameQuantifiers(label.substring(0, firstSpaceIndex)); - let attribute = { + result = result.toString().replace(/\s+/g, " "); + const firstSpaceIndex = result[0] == columnQuantifiers.Start && + result.indexOf(columnQuantifiers.End + " ") !== -1 + ? result.indexOf(columnQuantifiers.End + " ") + : result.indexOf(" "); + const attributeType = result.substring(firstSpaceIndex + 1).trim(); + const attributeName = RemoveNameQuantifiers(result.substring(0, firstSpaceIndex + 1)); + const attribute = { attributeName, attributeType - } - return attribute + }; + return attribute; } function RemoveNameQuantifiers(name) { return name.replace(/\[|\]|\(|\"|\'|\`/g, "").trim(); } - - function getMermaidDiagramDb(type){ - var model = ui.editor.graph.getModel() + function getMermaidDiagramDb(type) { + const model = ui.editor.graph.getModel(); // same models from mermaid for diagram relationships // only difference is entities is an array rather than object to allow duplicate tables - let entities = {} - let relationships = [] + const entities = {}; + const relationships = []; // build models for (const key in model.cells) { if (Object.hasOwnProperty.call(model.cells, key)) { const mxcell = model.cells[key]; - if(mxcell.mxObjectId.indexOf("mxCell") !== -1) { - if(mxcell.style && mxcell.style.trim().startsWith("swimlane;")){ - let entity = { + if (mxcell.mxObjectId.indexOf("mxCell") !== -1) { + if (mxcell.style && mxcell.style.trim().startsWith("swimlane;")) { + const entity = { name: RemoveNameQuantifiers(mxcell.value), - attributes: [] - } + attributes: [], + }; for (let c = 0; c < mxcell.children.length; c++) { const col = mxcell.children[c]; - if(col.mxObjectId.indexOf("mxCell") !== -1) { - if(col.style && col.style.trim().startsWith("shape=partialRectangle")){ + if (col.mxObjectId.indexOf("mxCell") !== -1) { + if (col.style && col.style.trim().startsWith("shape=partialRectangle")) { const columnQuantifiers = GetColumnQuantifiers(type); //Get delimiter of column name //Get full name - let attribute = getDbLabel(col.value, columnQuantifiers) - var attributeKeyType = col.children.find(x=> ["FK","PK"].findIndex(k => k== x.value.toUpperCase()) !== -1 || - x.value.toUpperCase().indexOf("PK,")!=-1) - if(attributeKeyType){ - attribute.attributeKeyType = attributeKeyType.value - if(attribute.attributeKeyType != "PK" && attribute.attributeKeyType.indexOf("PK") != -1){ - attribute.attributeKeyType = "PK" + const attribute = getDbLabel(col.value, columnQuantifiers); + const attributeKeyType = col.children.find(x => ["FK", "PK"].findIndex(k => k == x.value.toUpperCase()) !== -1 || + x.value.toUpperCase().indexOf("PK,") != -1); + if (attributeKeyType) { + attribute.attributeKeyType = attributeKeyType.value; + if (attribute.attributeKeyType != "PK" && attribute.attributeKeyType.indexOf("PK") != -1) { + attribute.attributeKeyType = "PK"; } } - entity.attributes.push(attribute) - if(col.edges && col.edges.length){ + entity.attributes.push(attribute); + if (col.edges && col.edges.length) { // check for edges foreign keys for (let e = 0; e < col.edges.length; e++) { const edge = col.edges[e]; - if(edge.mxObjectId.indexOf("mxCell") !== -1) { - if(edge.style && edge.style.indexOf("endArrow=") != -1 && edge.source && - edge.source.value && edge.target && edge.target.value){ - // need to check if end is open or certain value to determin relationship type - // extract endArrow txt - // check if both match and contain many or open - // if both match and are many then create a new table - let endCheck = "endArrow=" - let endArr = edge.style.indexOf(endCheck) != -1 ? + if (edge.mxObjectId.indexOf("mxCell") !== -1) { + if (edge.style && edge.style.indexOf("endArrow=") != -1 && edge.source && + edge.source.value && edge.target && edge.target.value) { + // need to check if end is open or certain value to determin relationship type + // extract endArrow txt + // check if both match and contain many or open + // if both match and are many then create a new table + const endCheck = "endArrow="; + const endArr = edge.style.indexOf(endCheck) != -1 ? edge.style.substring(edge.style.indexOf(endCheck) + endCheck.length, edge.style.substring(edge.style.indexOf(endCheck) + endCheck.length).indexOf(";") + edge.style.indexOf(endCheck) + endCheck.length) - : "" - let startCheck = "startArrow=" - let startArr = edge.style.indexOf(startCheck) != -1 ? + : ""; + const startCheck = "startArrow="; + const startArr = edge.style.indexOf(startCheck) != -1 ? edge.style.substring(edge.style.indexOf(startCheck) + startCheck.length, edge.style.substring(edge.style.indexOf(startCheck) + startCheck.length).indexOf(";") + edge.style.indexOf(startCheck) + startCheck.length) - : "" - - var manyCheck = ["open","many"] - var sourceIsPrimary = endArr && manyCheck - .findIndex(x => endArr.toLocaleLowerCase().indexOf(x)!=-1) != -1; - var targetIsPrimary = startArr && manyCheck - .findIndex(x => startArr.toLocaleLowerCase().indexOf(x)!=-1) != -1; - // has to be one to many and not one to one - if((targetIsPrimary || sourceIsPrimary) && - !(targetIsPrimary && sourceIsPrimary) - ){ - var sourceId = edge.source.value; - sourceAttr = getDbLabel(sourceId, columnQuantifiers); - sourceId = sourceAttr.attributeName - var sourceEntity = RemoveNameQuantifiers(edge.source.parent.value); - var targetId = edge.target.value; - targetAttr = getDbLabel(targetId, columnQuantifiers); - targetId = targetAttr.attributeName - var targetEntity = RemoveNameQuantifiers(edge.target.parent.value); - // entityA primary - // entityB foreign - let relationship = { - entityA: sourceIsPrimary ? sourceEntity : targetEntity, - entityB: sourceIsPrimary ? targetEntity : sourceEntity, - // based off of styles? - relSpec: { - cardA: 'ZERO_OR_MORE', - cardB: 'ONLY_ONE', - relType: "IDENTIFYING" - }, - roleA: sourceIsPrimary ? - `[${sourceEntity}.${sourceId}] to [${targetEntity}.${targetId}]` : - `[${targetEntity}.${targetId}] to [${sourceEntity}.${sourceId}]` - } - // check that is doesn't already exist - var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA) - if(exists ==-1){ - relationships.push(relationship) - } - } else if(targetIsPrimary && sourceIsPrimary){ - // add a new many to many table - var sourceId = edge.source.value; - sourceAttr = getDbLabel(sourceId, columnQuantifiers); - sourceAttr.attributeKeyType = "PK" - sourceId = sourceAttr.attributeName - var sourceEntity = RemoveNameQuantifiers(edge.source.parent.value); - var targetId = edge.target.value; - targetAttr = getDbLabel(targetId, columnQuantifiers); - targetAttr.attributeKeyType = "PK" - targetId = targetAttr.attributeName - var targetEntity = RemoveNameQuantifiers(edge.target.parent.value); - let compositeEntity = { - name: RemoveNameQuantifiers(sourceEntity) + "_" + RemoveNameQuantifiers(targetEntity), - attributes: [sourceAttr, targetAttr] - } - // add composite entity - if(entities[compositeEntity.name]){ - // DON'T add duplicate composite tables - } else { - entities[compositeEntity.name] = compositeEntity - } - // entityA primary - // entityB foreign - let relationship = { - entityA: sourceEntity, - entityB: compositeEntity.name, - // based off of styles? - relSpec: { - cardA: 'ZERO_OR_MORE', - cardB: 'ONLY_ONE', - relType: "IDENTIFYING" - }, - roleA: `[${sourceEntity}.${sourceId}] to [${compositeEntity.name}.${sourceId}]` - } - // check that is doesn't already exist - var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA) - if(exists ==-1){ - relationships.push(relationship) - } - let relationship2 = { - entityA: targetEntity, - entityB: compositeEntity.name, - // based off of styles? - relSpec: { - cardA: 'ZERO_OR_MORE', - cardB: 'ONLY_ONE', - relType: "IDENTIFYING" - }, - roleA: `[${targetEntity}.${targetId}] to [${compositeEntity.name}.${targetId}]` - } - // check that is doesn't already exist - exists = relationships.findIndex(r => r.entityA == relationship2.entityA && r.entityB == relationship2.entityB && r.roleA == relationship2.roleA) - if(exists ==-1){ - relationships.push(relationship2) - } + : ""; + const manyCheck = ["open", "many"]; + const sourceIsPrimary = endArr && manyCheck + .findIndex(x => endArr.toLocaleLowerCase().indexOf(x) != -1) != -1; + const targetIsPrimary = startArr && manyCheck + .findIndex(x => startArr.toLocaleLowerCase().indexOf(x) != -1) != -1; + // has to be one to many and not one to one + if ((targetIsPrimary || sourceIsPrimary) && + !(targetIsPrimary && sourceIsPrimary)) { + var sourceId = edge.source.value; + var sourceAttr = getDbLabel(sourceId, columnQuantifiers); + sourceId = sourceAttr.attributeName; + var sourceEntity = RemoveNameQuantifiers(edge.source.parent.value); + var targetId = edge.target.value; + var targetAttr = getDbLabel(targetId, columnQuantifiers); + targetId = targetAttr.attributeName; + var targetEntity = RemoveNameQuantifiers(edge.target.parent.value); + // entityA primary + // entityB foreign + const relationship = { + entityA: sourceIsPrimary ? sourceEntity : targetEntity, + entityB: sourceIsPrimary ? targetEntity : sourceEntity, + // based off of styles? + relSpec: { + cardA: "ZERO_OR_MORE", + cardB: "ONLY_ONE", + relType: "IDENTIFYING" + }, + roleA: sourceIsPrimary ? + `[${sourceEntity}.${sourceId}] to [${targetEntity}.${targetId}]` : + `[${targetEntity}.${targetId}] to [${sourceEntity}.${sourceId}]` + }; + // check that is doesn't already exist + var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA); + if (exists == -1) { + relationships.push(relationship); } - + } + else if (targetIsPrimary && sourceIsPrimary) { + // add a new many to many table + var sourceId = edge.source.value; + sourceAttr = getDbLabel(sourceId, columnQuantifiers); + sourceAttr.attributeKeyType = "PK"; + sourceId = sourceAttr.attributeName; + var sourceEntity = RemoveNameQuantifiers(edge.source.parent.value); + var targetId = edge.target.value; + targetAttr = getDbLabel(targetId, columnQuantifiers); + targetAttr.attributeKeyType = "PK"; + targetId = targetAttr.attributeName; + var targetEntity = RemoveNameQuantifiers(edge.target.parent.value); + const compositeEntity = { + name: RemoveNameQuantifiers(sourceEntity) + "_" + RemoveNameQuantifiers(targetEntity), + attributes: [sourceAttr, targetAttr] + }; + // add composite entity + if (entities[compositeEntity.name]) { + // DON'T add duplicate composite tables + } + else { + entities[compositeEntity.name] = compositeEntity; + } + // entityA primary + // entityB foreign + const relationship = { + entityA: sourceEntity, + entityB: compositeEntity.name, + // based off of styles? + relSpec: { + cardA: "ZERO_OR_MORE", + cardB: "ONLY_ONE", + relType: "IDENTIFYING" + }, + roleA: `[${sourceEntity}.${sourceId}] to [${compositeEntity.name}.${sourceId}]` + }; + // check that is doesn't already exist + var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA); + if (exists == -1) { + relationships.push(relationship); + } + const relationship2 = { + entityA: targetEntity, + entityB: compositeEntity.name, + // based off of styles? + relSpec: { + cardA: "ZERO_OR_MORE", + cardB: "ONLY_ONE", + relType: "IDENTIFYING" + }, + roleA: `[${targetEntity}.${targetId}] to [${compositeEntity.name}.${targetId}]` + }; + // check that is doesn't already exist + exists = relationships.findIndex(r => r.entityA == relationship2.entityA && r.entityB == relationship2.entityB && r.roleA == relationship2.roleA); + if (exists == -1) { + relationships.push(relationship2); + } + } } } - } } } } } // allows for duplicates if another table has the same name - if(entities[entity.name]){ - var count = 2; - while(entities[entity.name + count.toString()]){ + if (entities[entity.name]) { + let count = 2; + while (entities[entity.name + count.toString()]) { count++; } - entities[entity.name + count.toString()] = entity - } else { - entities[entity.name] = entity + entities[entity.name + count.toString()] = entity; + } + else { + entities[entity.name] = entity; } } - } } } - - class DbDefinition{ - constructor(entities, relationships){ + class DatabaseModel { + constructor(entities, relationships) { this.entities = entities; this.relationships = relationships; } - - getEntities(){ - return this.entities + getEntities() { + return this.entities; } - - getRelationships(){ - return this.relationships + getRelationships() { + return this.relationships; } } - - var db = new DbDefinition(entities, relationships); - + const db = new DatabaseModel(entities, relationships); return db; } - function generateSql(type) { - // get diagram model - var db = getMermaidDiagramDb(type); + const db = getMermaidDiagramDb(type); // load parser - var parser = new DbParser(type, db) + const parser = new generate_sql_ddl_1.DbParser(type, db); // generate sql - var sql = parser.getSQLDataDefinition() - sql = `/*\n\tGenerated in drawio\n\tDatabase: ${type}\n*/\n\n` + sql + let sql = parser.getSQLDataDefinition(); + sql = `/*\n\tGenerated in drawio\n\tDatabase: ${type}\n*/\n\n` + sql; sql = sql.trim(); // update sql value in text area sqlInputGenSQL.value = sql; // TODO: use selection as well? - var modelSelected = ui.editor.graph.getSelectionModel() - }; - + const modelSelected = ui.editor.graph.getSelectionModel(); + } + ; mxUtils.br(divGenSQL); - - const resetBtnGenSQL = mxUtils.button(mxResources.get('reset'), function() { - sqlInputGenSQL.value = ''; + const resetBtnGenSQL = mxUtils.button(mxResources.get("reset"), function () { + sqlInputGenSQL.value = ""; }); - - resetBtnGenSQL.style.marginTop = '8px'; - resetBtnGenSQL.style.marginRight = '4px'; - resetBtnGenSQL.style.padding = '4px'; + resetBtnGenSQL.style.marginTop = "8px"; + resetBtnGenSQL.style.marginRight = "4px"; + resetBtnGenSQL.style.padding = "4px"; divGenSQL.appendChild(resetBtnGenSQL); - - const btnGenSQL_mysql = mxUtils.button('MySQL', function() { - generateSql('mysql'); + const btnGenSQL_mysql = mxUtils.button("MySQL", function () { + generateSql("mysql"); }); - - btnGenSQL_mysql.style.marginTop = '8px'; - btnGenSQL_mysql.style.padding = '4px'; + btnGenSQL_mysql.style.marginTop = "8px"; + btnGenSQL_mysql.style.padding = "4px"; divGenSQL.appendChild(btnGenSQL_mysql); - - const btnGenSQL_sqlserver = mxUtils.button('SQL Server', function() { - generateSql('sqlserver'); + const btnGenSQL_sqlserver = mxUtils.button("SQL Server", function () { + generateSql("sqlserver"); }); - - btnGenSQL_sqlserver.style.marginTop = '8px'; - btnGenSQL_sqlserver.style.padding = '4px'; + btnGenSQL_sqlserver.style.marginTop = "8px"; + btnGenSQL_sqlserver.style.padding = "4px"; divGenSQL.appendChild(btnGenSQL_sqlserver); - - const btnGenSQL_postgres = mxUtils.button('PostgreSQL', function() { - generateSql('postgres'); + const btnGenSQL_postgres = mxUtils.button("PostgreSQL", function () { + generateSql("postgres"); }); - - btnGenSQL_postgres.style.marginTop = '8px'; - btnGenSQL_postgres.style.padding = '4px'; + btnGenSQL_postgres.style.marginTop = "8px"; + btnGenSQL_postgres.style.padding = "4px"; divGenSQL.appendChild(btnGenSQL_postgres); - - const btnGenSQL_sqlite = mxUtils.button('Sqlite', function() { - generateSql('sqlite'); + const btnGenSQL_sqlite = mxUtils.button("Sqlite", function () { + generateSql("sqlite"); }); - - btnGenSQL_sqlite.style.marginTop = '8px'; - btnGenSQL_sqlite.style.padding = '4px'; + btnGenSQL_sqlite.style.marginTop = "8px"; + btnGenSQL_sqlite.style.padding = "4px"; divGenSQL.appendChild(btnGenSQL_sqlite); - // Adds action - ui.actions.addAction('tosql', function() { + ui.actions.addAction("tosql", function () { wndGenSQL.setVisible(!wndGenSQL.isVisible()); - if (wndGenSQL.isVisible()) { sqlInputGenSQL.focus(); } }); // end export sql methods - // import diagrams from sql text methods - - const contants_1 ={} - contants_1.CONSTRAINT_Foreign_Key = contants_1.CONSTRAINT_Primary_Key = contants_1.Foreign_Key = contants_1.Primary_Key = contants_1.CONSTRAINT = contants_1.AlterTable = contants_1.CreateTable = void 0; - contants_1.CreateTable = "create table"; - contants_1.AlterTable = "alter table "; - contants_1.CONSTRAINT = "constraint"; - contants_1.Primary_Key = "primary key"; - contants_1.Foreign_Key = "foreign key"; - contants_1.CONSTRAINT_Primary_Key = "constraint primary key"; - contants_1.CONSTRAINT_Foreign_Key = "constraint foreign key"; - - /** - * Main Parser class - */ - class SqlSimpleParser { - /** - * Parser constructor. - * Default dialect is 'sqlite'. - * Options: "mysql" | "sqlite" | "postgres" | "sqlserver" . - * - * @param dialect SQL dialect ('sqlite'). - */ - constructor(dialect = "sqlite") { - this.tableList = []; - /** - * Parsed statements. - */ - this.statements = []; - /** - * Remains of string feed, after last parsed statement. - */ - this.remains = ""; - /** - * Whether preparser is currently escaped. - */ - this.escaped = false; - /** - * Current quote char of preparser. - */ - this.quoted = ""; - this.exportedTables = 0; - this.SQLServer = "sqlserver"; - this.MODE_SQLSERVER = false; - this.foreignKeyList = []; - this.primaryKeyList = []; - this.dialect = dialect; - this.MODE_SQLSERVER = - this.dialect !== undefined && - this.dialect !== null && - this.dialect == this.SQLServer; - } - /** - * Feed chunk of string into parser. - * - * @param chunk Chunk of string to be parsed. - */ - feed(chunk) { - // - const removedComments = chunk - // remove database comments, multiline, --, and // - .replace(/\/\*[\s\S]*?\*\/|\/\/|--.*/g, "") - .replace(/IF NOT EXISTS/gi, "") - .trim(); - const cleanedLines = removedComments - .split("\n") - // remove empty lines - .filter((n) => n) - // remove multiple spaces - .map((n) => n.replace(/\s+/g, " ").trim()); - // combine lines that are in parenthesis - const lines = []; - let insertSameLine = false; - cleanedLines.forEach((n) => { - if (lines.length > 0){ - if((n[0] == "(" && - lines[lines.length - 1].toLocaleLowerCase().indexOf(contants_1.CreateTable) == - -1) || - insertSameLine) { - if (lines.length > 0) { - insertSameLine = true; - lines[lines.length - 1] += ` ${n}`; - if (n[0] == ")") - insertSameLine = false; - } - } - else if(lines[lines.length - 1].match(/CONSTRAINT/gi) && - (n.match(/FOREIGN KEY/gi) && !n.match(/CONSTRAINT/gi)) - ){ - lines[lines.length - 1] += ` ${n}`; - } - // add to previous line if current has references and previous has foreign key - else if(lines[lines.length - 1].match(/FOREIGN KEY/gi) && - (n.match(/REFERENCES/gi) && !n.match(/FOREIGN KEY/gi)) - ){ - lines[lines.length - 1] += ` ${n}`; - } - else if(n.substring(0,2).toUpperCase() == "ON"){ - lines[lines.length - 1] += ` ${n}`; - } - else { - lines.push(n); - } - } - else { - lines.push(n); - } - }); - let currentTableModel = null; - //Parse SQL to objects - for (let i = 0; i < lines.length; i++) { - // rowCell = null; - const tmp = lines[i].trim(); - const propertyRow = tmp.toLowerCase().trim(); - if (propertyRow[0] == ")") { - // close table - if (currentTableModel) { - this.tableList.push(currentTableModel); - currentTableModel = null; - } - continue; - } - //Parse Table - if (propertyRow.indexOf(contants_1.CreateTable) != -1) { - //Parse row - let name = tmp - .replace(this.stringToRegex(`/${contants_1.CreateTable}/gi`), "") - .trim(); - //Parse Table Name - name = this.ParseTableName(name); - if (currentTableModel !== null) { - //Add table to the list - this.tableList.push(currentTableModel); - } - //Create Table - currentTableModel = this.CreateTable(name); - } - // Parse Properties - else if (tmp !== "(" && - currentTableModel != null && - propertyRow.indexOf(contants_1.AlterTable) == -1) { - //Parse the row - let name = tmp.substring(0, tmp.charAt(tmp.length - 1) === "," ? tmp.length - 1 : tmp.length); - //Attempt to get the Key Type - let propertyType = name.toLowerCase(); - // .substring(0, AlterTable.length).toLowerCase(); - //Add special constraints - if (this.MODE_SQLSERVER) { - if (propertyType.indexOf(contants_1.CONSTRAINT) !== -1 && - propertyType.indexOf(contants_1.Primary_Key) !== -1) { - propertyType = contants_1.CONSTRAINT_Primary_Key; - } - if (propertyType.indexOf(contants_1.CONSTRAINT) !== -1 && - propertyType.indexOf(contants_1.Foreign_Key) !== -1) { - propertyType = contants_1.CONSTRAINT_Foreign_Key; - } - } - //Verify if this is a property that doesn't have a relationship (One minute of silence for the property) - const normalProperty = !propertyType.match(/PRIMARY KEY\s?\(/gi) && - propertyType.indexOf(contants_1.Foreign_Key) == -1 && - propertyType.indexOf(contants_1.CONSTRAINT_Primary_Key) == -1 && - propertyType.indexOf(contants_1.CONSTRAINT_Foreign_Key) == -1; - const nameSkipCheck = name.toUpperCase().trim(); - //Parse properties that don't have relationships - if (normalProperty) { - if (name === "" || name === "" || name === ");") { - continue; - } - let ExtendedProperties = null; - if (this.MODE_SQLSERVER) { - if (nameSkipCheck.indexOf(" ASC") !== -1 || - nameSkipCheck.indexOf(" DESC") !== -1 || - nameSkipCheck.indexOf("EXEC ") !== -1 || - nameSkipCheck.indexOf("WITH ") !== -1 || - nameSkipCheck.indexOf(" ON") !== -1 || - nameSkipCheck.indexOf("ALTER ") !== -1 || - // comments already removed - nameSkipCheck.indexOf("/*") !== -1 || - nameSkipCheck.indexOf(" CONSTRAINT") !== -1 || - nameSkipCheck.indexOf("SET ") !== -1 || - nameSkipCheck.indexOf(" NONCLUSTERED") !== -1 || - // no spaces desired - nameSkipCheck.indexOf("GO") !== -1 || - nameSkipCheck.indexOf("REFERENCES ") !== -1) { - continue; - } - //Get delimiter of column name - const firstSpaceIndex = name[0] == "[" && name.indexOf("]" + " ") !== -1 - ? name.indexOf("]" + " ") - : name.indexOf(" "); - ExtendedProperties = name.substring(firstSpaceIndex + 1).trim(); - //Get full name - name = name.substring(0, firstSpaceIndex); - name = this.RemoveNameQuantifiers(name); - } - else { - const columnQuantifiers = this.GetColumnQuantifiers(); - //Get delimiter of column name - const firstSpaceIndex = name[0] == columnQuantifiers.Start && - name.indexOf(columnQuantifiers.End + " ") !== -1 - ? name.indexOf(columnQuantifiers.End + " ") - : name.indexOf(" "); - ExtendedProperties = name.substring(firstSpaceIndex + 1).trim(); - //Get full name - name = name.substring(0, firstSpaceIndex); - name = this.RemoveNameQuantifiers(name); - } - //Create Property - const propertyModel = this.CreateProperty(name, currentTableModel.Name, null, false, ExtendedProperties); - //Add Property to table - currentTableModel.Properties.push(propertyModel); - if (ExtendedProperties.toLocaleLowerCase().indexOf(contants_1.Primary_Key) > -1) { - //Create Primary Key - const primaryKeyModel = this.CreatePrimaryKey(name, currentTableModel.Name); - //Add Primary Key to List - this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); - } - } - else { - //Parse Primary Key - if (propertyType.indexOf(contants_1.Primary_Key) != -1 || - propertyType.indexOf(contants_1.CONSTRAINT_Primary_Key) != -1) { - if (!this.MODE_SQLSERVER) { - const primaryKey = name - .replace(/PRIMARY KEY\s?\(/gi, "") - .replace(")", ""); - //Create Primary Key - const primaryKeyModel = this.CreatePrimaryKey(primaryKey, currentTableModel.Name); - //Add Primary Key to List - this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); - } - else { - if (propertyRow.indexOf(contants_1.Primary_Key) !== -1 && - nameSkipCheck.indexOf("CLUSTERED") === -1) { - const primaryKey = name - .replace(/PRIMARY KEY\s?\(/gi, "") - .replace(")", ""); - //Create Primary Key - const primaryKeyModel = this.CreatePrimaryKey(primaryKey, currentTableModel.Name); - //Add Primary Key to List - this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); - } - else { - const startIndex = name.toLocaleLowerCase().indexOf("("); - const endIndex = name.indexOf(")") + 1; - const primaryKey = name - .substring(startIndex, endIndex) - .replace("(", "") - .replace(")", "") - .replace(/ASC/gi, "") - .trim(); - const columnQuantifiers = this.GetColumnQuantifiers(); - //Get delimiter of column name - const firstSpaceIndex = primaryKey[0] == columnQuantifiers.Start && - primaryKey.indexOf(columnQuantifiers.End + " ") !== -1 - ? primaryKey.indexOf(columnQuantifiers.End + " ") - : primaryKey.indexOf(" "); - const primaryKeyRow = firstSpaceIndex == -1 - ? primaryKey - : primaryKey.substring(firstSpaceIndex + 1).trim(); - //Create Primary Key - const primaryKeyModel = this.CreatePrimaryKey(primaryKeyRow, currentTableModel.Name); - //Add Primary Key to List - this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); - } - } - } - } - //Parse Foreign Key - if (propertyType.indexOf(contants_1.Foreign_Key) != -1 || - propertyType.indexOf(contants_1.CONSTRAINT_Foreign_Key) != -1) { - if (!this.MODE_SQLSERVER || propertyRow.indexOf(contants_1.AlterTable) == -1) { - this.ParseMySQLForeignKey(name, currentTableModel); - } - else { - let completeRow = name; - if (nameSkipCheck.indexOf("REFERENCES") === -1) { - const referencesRow = lines[i + 1].trim(); - completeRow = - "ALTER TABLE [dbo].[" + - currentTableModel.Name + - "] WITH CHECK ADD" + - " " + - name + - " " + - referencesRow; - } - this.ParseSQLServerForeignKey(completeRow, currentTableModel); - } - } - } - else if (propertyRow.indexOf(contants_1.AlterTable) != -1) { - if (this.MODE_SQLSERVER) { - //Parse the row - const alterTableRow = tmp.substring(0, tmp.charAt(tmp.length - 1) === "," ? tmp.length - 1 : tmp.length); - const referencesRow = lines[i + 1].trim(); - const completeRow = alterTableRow + " " + referencesRow; - this.ParseSQLServerForeignKey(completeRow, currentTableModel); - } - } - } - // parse fk and primary keys - if (this.primaryKeyList.length > 0) { - this.primaryKeyList.forEach((pk) => { - // find table index - const pkTableIndex = this.tableList.findIndex((t) => t.Name.toLocaleLowerCase() == - pk.PrimaryKeyTableName.toLocaleLowerCase()); - // find property index - if (pkTableIndex > -1) { - const propertyIndex = this.tableList[pkTableIndex].Properties.findIndex((p) => p.Name.toLocaleLowerCase() == - pk.PrimaryKeyName.toLocaleLowerCase()); - if (propertyIndex > -1) { - this.tableList[pkTableIndex].Properties[propertyIndex].IsPrimaryKey = true; - } - } - }); - } - if (this.foreignKeyList.length > 0) { - this.foreignKeyList.forEach((fk) => { - // find table index - const pkTableIndex = this.tableList.findIndex((t) => t.Name.toLocaleLowerCase() == - fk.ReferencesTableName.toLocaleLowerCase()); - // find property index - if (pkTableIndex > -1) { - const propertyIndex = this.tableList[pkTableIndex].Properties.findIndex((p) => p.Name.toLocaleLowerCase() == - fk.PrimaryKeyName.toLocaleLowerCase()); - if (propertyIndex > -1) { - this.tableList[pkTableIndex].Properties[propertyIndex].ForeignKey.push(fk); - if (!fk.IsDestination) { - this.tableList[pkTableIndex].Properties[propertyIndex].IsForeignKey = true; - } - } - } - }); - } - return this; - } - stringToRegex(str) { - // Main regex - const mainResult = str.match(/\/(.+)\/.*/); - const optionsResult = str.match(/\/.+\/(.*)/); - if (mainResult && optionsResult) { - const main = mainResult[1]; - // Regex options - const options = optionsResult[1]; - // Compiled regex - return new RegExp(main, options); - } - return new RegExp("//(.+)/.*/", "//.+/(.*)/"); - } - CreatePrimaryKey(primaryKeyName, primaryKeyTableName) { - const primaryKeyNames = this.RemoveNameQuantifiers(primaryKeyName) - .split(",") - .filter((n) => n) - // remove multiple spaces - .map((n) => n.replace(/\s+/g, " ").trim()); - const primaryKeys = []; - primaryKeyNames.forEach(name => { - const primaryKey = { - PrimaryKeyTableName: primaryKeyTableName, - PrimaryKeyName: name, - }; - primaryKeys.push(primaryKey); - }); - return primaryKeys; - } - CreateProperty(name, tableName, foreignKey, isPrimaryKey, columnProps) { - const isForeignKey = foreignKey !== undefined && foreignKey !== null; - const property = { - Name: name, - ColumnProperties: columnProps, - TableName: tableName, - ForeignKey: foreignKey || [], - IsForeignKey: isForeignKey, - IsPrimaryKey: isPrimaryKey, - }; - return property; - } - ParseMySQLForeignKey(name, currentTableModel) { - const referencesIndex = name.toLowerCase().indexOf("references"); - let foreignKeySQL = name.substring(0, referencesIndex); - foreignKeySQL = foreignKeySQL.substring(foreignKeySQL.toUpperCase().indexOf("FOREIGN KEY")) - let referencesSQL = name.substring(referencesIndex, name.length); - //Remove references syntax - referencesSQL = referencesSQL.replace(/REFERENCES /gi, ""); - //Get Table and Property Index - const referencedTableIndex = referencesSQL.indexOf("("); - const referencedPropertyIndex = referencesSQL.indexOf(")"); - //Get Referenced Table - const referencedTableName = referencesSQL.substring(0, referencedTableIndex); - //Get Referenced Key - const referencedPropertyName = referencesSQL.substring(referencedTableIndex + 1, referencedPropertyIndex); - //Get ForeignKey - const foreignKey = foreignKeySQL - .replace(/FOREIGN KEY\s?\(/gi, "") - .replace(")", "") - .replace(" ", ""); - //Create ForeignKey - const foreignKeyOriginModel = this.CreateForeignKey(foreignKey, currentTableModel.Name, referencedPropertyName, referencedTableName, true); - //Add ForeignKey Origin - this.foreignKeyList.push(foreignKeyOriginModel); - //Create ForeignKey - const foreignKeyDestinationModel = this.CreateForeignKey(referencedPropertyName, referencedTableName, foreignKey, currentTableModel.Name, false); - //Add ForeignKey Destination - this.foreignKeyList.push(foreignKeyDestinationModel); - } - ParseSQLServerForeignKey(name, currentTableModel) { - const referencesIndex = name.toLowerCase().indexOf("references"); - let foreignKeySQL = ""; - if (name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`) !== -1) { - foreignKeySQL = name - .substring(name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`), referencesIndex) - .replace(/FOREIGN KEY\(/gi, "") - .replace(")", ""); - } - else { - foreignKeySQL = name - .substring(name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`), referencesIndex) - .replace(/FOREIGN KEY\s?\(/gi, "") - .replace(")", ""); - } - let referencesSQL = name.substring(referencesIndex, name.length); - const nameSkipCheck = name.toUpperCase().trim(); - let alterTableName = name - .substring(0, nameSkipCheck.indexOf("WITH")) - .replace(/ALTER TABLE /gi, ""); - if (referencesIndex !== -1 && - alterTableName !== "" && - foreignKeySQL !== "" && - referencesSQL !== "") { - //Remove references syntax - referencesSQL = referencesSQL.replace(/REFERENCES /gi, ""); - //Get Table and Property Index - const referencedTableIndex = referencesSQL.indexOf("("); - const referencedPropertyIndex = referencesSQL.indexOf(")"); - //Get Referenced Table - let referencedTableName = referencesSQL.substring(0, referencedTableIndex); - //Parse Name - referencedTableName = this.ParseSQLServerName(referencedTableName); - //Get Referenced Key - let referencedPropertyName = referencesSQL.substring(referencedTableIndex + 1, referencedPropertyIndex); - //Parse Name - referencedPropertyName = this.ParseSQLServerName(referencedPropertyName); - //Get ForeignKey - let foreignKey = foreignKeySQL - .replace(/FOREIGN KEY\s?\(/gi, "") - .replace(")", ""); - //Parse Name - foreignKey = this.ParseSQLServerName(foreignKey); - //Parse Name - alterTableName = this.ParseSQLServerName(alterTableName); - //Create ForeignKey - const foreignKeyOriginModel = this.CreateForeignKey(foreignKey, alterTableName, referencedPropertyName, referencedTableName, true); - //Add ForeignKey Origin - this.foreignKeyList.push(foreignKeyOriginModel); - //Create ForeignKey - const foreignKeyDestinationModel = this.CreateForeignKey(referencedPropertyName, referencedTableName, foreignKey, alterTableName, false); - //Add ForeignKey Destination - this.foreignKeyList.push(foreignKeyDestinationModel); - } - } - CreateForeignKey(primaryKeyName, primaryKeyTableName, referencesPropertyName, referencesTableName, isDestination) { - const foreignKey = { - PrimaryKeyTableName: this.RemoveNameQuantifiers(primaryKeyTableName), - PrimaryKeyName: this.RemoveNameQuantifiers(primaryKeyName), - ReferencesPropertyName: this.RemoveNameQuantifiers(referencesPropertyName), - ReferencesTableName: this.RemoveNameQuantifiers(referencesTableName), - IsDestination: isDestination !== undefined && isDestination !== null - ? isDestination - : false, - }; - return foreignKey; - } - RemoveNameQuantifiers(name) { - return name.replace(/\[|\]|\(|\"|\'|\`/g, "").trim(); - } - ParseTableName(name) { - if (name.charAt(name.length - 1) === "(") { - name = this.RemoveNameQuantifiers(name); - } - return name; - } - ParseSQLServerName(name, property) { - name = name.replace("[dbo].[", ""); - name = name.replace("](", ""); - name = name.replace("].[", "."); - name = name.replace("[", ""); - if (property == undefined || property == null) { - name = name.replace(" [", ""); - name = name.replace("] ", ""); - } - else { - if (name.indexOf("]") !== -1) { - name = name.substring(0, name.indexOf("]")); - } - } - if (name.lastIndexOf("]") === name.length - 1) { - name = name.substring(0, name.length - 1); - } - if (name.lastIndexOf(")") === name.length - 1) { - name = name.substring(0, name.length - 1); - } - if (name.lastIndexOf("(") === name.length - 1) { - name = name.substring(0, name.length - 1); - } - name = name.replace(" ", ""); - return name; - } - CreateTable(name) { - const table = { - Name: name, - Properties: [], - }; - //Count exported tables - this.exportedTables++; - return table; - } - /** - * Checks whether character is a quotation character. - * - * @param char Character to be evaluated. - */ - static isQuoteChar(char) { - return char === "\"" || char === "'" || char === "`"; - } - /** - * convert labels with start and end strings per database type - * @param label - * @returns - */ - dbTypeEnds(label) { - let char1 = "\""; - let char2 = "\""; - if (this.dialect == "mysql") { - char1 = "`"; - char2 = "`"; - } - else if (this.dialect == "sqlserver") { - char1 = "["; - char2 = "]"; - } - return `${char1}${label}${char2}`; - } - WithEnds() { - this.tableList = this.tableList.map((table) => { - table.Name = this.dbTypeEnds(table.Name); - table.Properties = table.Properties.map((property) => { - property.Name = this.dbTypeEnds(property.Name); - property.TableName = this.dbTypeEnds(property.TableName); - return property; - }); - return table; - }); - this.primaryKeyList = this.primaryKeyList.map((primaryKey) => { - primaryKey.PrimaryKeyName = this.dbTypeEnds(primaryKey.PrimaryKeyName); - primaryKey.PrimaryKeyTableName = this.dbTypeEnds(primaryKey.PrimaryKeyTableName); - return primaryKey; - }); - this.foreignKeyList = this.foreignKeyList.map((foreignKey) => { - foreignKey.PrimaryKeyName = this.dbTypeEnds(foreignKey.PrimaryKeyName); - foreignKey.ReferencesPropertyName = this.dbTypeEnds(foreignKey.ReferencesPropertyName); - foreignKey.PrimaryKeyTableName = this.dbTypeEnds(foreignKey.PrimaryKeyTableName); - foreignKey.ReferencesTableName = this.dbTypeEnds(foreignKey.ReferencesTableName); - return foreignKey; - }); - return this; - } - WithoutEnds() { - this.tableList.map((table) => { - table.Name = this.RemoveNameQuantifiers(table.Name); - table.Properties = table.Properties.map((property) => { - property.Name = this.RemoveNameQuantifiers(property.Name); - property.TableName = this.RemoveNameQuantifiers(property.TableName); - return property; - }); - return table; - }); - this.primaryKeyList = this.primaryKeyList.map((primaryKey) => { - primaryKey.PrimaryKeyName = this.RemoveNameQuantifiers(primaryKey.PrimaryKeyName); - primaryKey.PrimaryKeyTableName = this.RemoveNameQuantifiers(primaryKey.PrimaryKeyTableName); - return primaryKey; - }); - this.foreignKeyList = this.foreignKeyList.map((foreignKey) => { - foreignKey.PrimaryKeyName = this.RemoveNameQuantifiers(foreignKey.PrimaryKeyName); - foreignKey.ReferencesPropertyName = this.RemoveNameQuantifiers(foreignKey.ReferencesPropertyName); - foreignKey.PrimaryKeyTableName = this.RemoveNameQuantifiers(foreignKey.PrimaryKeyTableName); - foreignKey.ReferencesTableName = this.RemoveNameQuantifiers(foreignKey.ReferencesTableName); - return foreignKey; - }); - return this; - } - /** - * return text quantifiers for dialect - * @returns json - */ - GetColumnQuantifiers() { - const chars = { - Start: "\"", - End: "\"", - }; - if (this.dialect == "mysql") { - chars.Start = "`"; - chars.End = "`"; - } - else if (this.dialect == "sqlserver") { - chars.Start = "["; - chars.End = "]"; - } - return chars; - } - ToTableList() { - return this.tableList; - } - ToPrimaryKeyList() { - return this.primaryKeyList; - } - ToForeignKeyList() { - return this.foreignKeyList; - } - ResetModel() { - this.tableList = []; - this.primaryKeyList = []; - this.foreignKeyList = []; - return this; - } - /** - * return full sql model - * @returns - */ - ToModel() { - return { - TableList: this.tableList, - Dialect: this.dialect, - ForeignKeyList: this.foreignKeyList, - PrimaryKeyList: this.primaryKeyList, - }; - } - } - //Table Info - var foreignKeyList = []; - var primaryKeyList = []; - var tableList = []; - var cells = []; - var tableCell = null; - var rowCell = null; - var dx = 0; - var exportedTables = 0; - - + let foreignKeyList = []; + let primaryKeyList = []; + let tableList = []; + let cells = []; + let tableCell = null; + let rowCell = null; + let dx = 0; + let exportedTables = 0; //Create Base div - const divFromSQL = document.createElement('div'); - divFromSQL.style.userSelect = 'none'; - divFromSQL.style.overflow = 'hidden'; - divFromSQL.style.padding = '10px'; - divFromSQL.style.height = '100%'; - + const divFromSQL = document.createElement("div"); + divFromSQL.style.userSelect = "none"; + divFromSQL.style.overflow = "hidden"; + divFromSQL.style.padding = "10px"; + divFromSQL.style.height = "100%"; var graph = ui.editor.graph; - - const sqlInputFromSQL = document.createElement('textarea'); - sqlInputFromSQL.style.height = '200px'; - sqlInputFromSQL.style.width = '100%'; - var defaultReset = '/*\n\tDraw io default value\n*/\n\nCREATE TABLE Persons\n(\n PersonID int NOT NULL,\n LastName varchar(255),\n ' + - 'FirstName varchar(255),\n Address varchar(255),\n City varchar(255),\n Primary Key(PersonID)\n);\n\n' + - 'CREATE TABLE Orders\n(\n OrderID int NOT NULL PRIMARY KEY,\n PersonID int NOT NULL,\n FOREIGN KEY ([PersonID]) REFERENCES [Persons]([PersonID])' + - '\n);' - sqlInputFromSQL.value = defaultReset + const sqlInputFromSQL = document.createElement("textarea"); + sqlInputFromSQL.style.height = "200px"; + sqlInputFromSQL.style.width = "100%"; + const defaultReset = "/*\n\tDraw io default value\n*/\n\nCREATE TABLE Persons\n(\n PersonID int NOT NULL,\n LastName varchar(255),\n " + + "FirstName varchar(255),\n Address varchar(255),\n City varchar(255),\n Primary Key(PersonID)\n);\n\n" + + "CREATE TABLE Orders\n(\n OrderID int NOT NULL PRIMARY KEY,\n PersonID int NOT NULL,\n FOREIGN KEY ([PersonID]) REFERENCES [Persons]([PersonID])" + + "\n);"; + sqlInputFromSQL.value = defaultReset; mxUtils.br(divFromSQL); divFromSQL.appendChild(sqlInputFromSQL); - var graph = ui.editor.graph; - // Extends Extras menu - mxResources.parse('fromSql=From SQL'); - - const wndFromSQL = new mxWindow(mxResources.get('fromSql'), divFromSQL, document.body.offsetWidth - 480, 140, - 320, 320, true, true); + mxResources.parse("fromSql=From SQL"); + const wndFromSQL = new mxWindow(mxResources.get("fromSql"), divFromSQL, document.body.offsetWidth - 480, 140, 320, 320, true, true); wndFromSQL.destroyOnClose = false; wndFromSQL.setMaximizable(false); wndFromSQL.setResizable(false); wndFromSQL.setClosable(true); - function AddRow(propertyModel, tableName) { - - var cellName = propertyModel.Name + (propertyModel.ColumnProperties ? " " + propertyModel.ColumnProperties: ""); - - rowCell = new mxCell(cellName, new mxGeometry(0, 0, 90, 26), - 'shape=partialRectangle;top=0;left=0;right=0;bottom=0;align=left;verticalAlign=top;spacingTop=-2;fillColor=none;spacingLeft=64;spacingRight=4;overflow=hidden;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;dropTarget=0;'); + const cellName = propertyModel.Name + (propertyModel.ColumnProperties ? " " + propertyModel.ColumnProperties : ""); + rowCell = new mxCell(cellName, new mxGeometry(0, 0, 90, 26), "shape=partialRectangle;top=0;left=0;right=0;bottom=0;align=left;verticalAlign=top;spacingTop=-2;fillColor=none;spacingLeft=64;spacingRight=4;overflow=hidden;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;dropTarget=0;"); rowCell.vertex = true; - - var columnType = propertyModel.IsPrimaryKey && propertyModel.IsForeignKey ? 'PK | FK' : propertyModel.IsPrimaryKey ? 'PK' : propertyModel.IsForeignKey ? 'FK' : ''; - - var left = sb.cloneCell(rowCell, columnType); + const columnType = propertyModel.IsPrimaryKey && propertyModel.IsForeignKey ? "PK | FK" : propertyModel.IsPrimaryKey ? "PK" : propertyModel.IsForeignKey ? "FK" : ""; + const left = sb.cloneCell(rowCell, columnType); left.connectable = false; - left.style = 'shape=partialRectangle;top=0;left=0;bottom=0;fillColor=none;align=left;verticalAlign=middle;spacingLeft=4;spacingRight=4;overflow=hidden;rotatable=180;points=[];portConstraint=eastwest;part=1;' + left.style = "shape=partialRectangle;top=0;left=0;bottom=0;fillColor=none;align=left;verticalAlign=middle;spacingLeft=4;spacingRight=4;overflow=hidden;rotatable=180;points=[];portConstraint=eastwest;part=1;"; left.geometry.width = 54; left.geometry.height = 26; rowCell.insert(left); - - var size = ui.editor.graph.getPreferredSizeForCell(rowCell); - - if (size !== null && tableCell.geometry.width < size.width + 10) { - tableCell.geometry.width = size.width + 10; + const size = ui.editor.graph.getPreferredSizeForCell(rowCell); + if (tableCell) { + if (size !== null && tableCell.geometry.width < size.width + 10) { + tableCell.geometry.width = size.width + 10; + } + tableCell.insert(rowCell); + tableCell.geometry.height += 26; } - - tableCell.insert(rowCell); - tableCell.geometry.height += 26; - rowCell = rowCell; - - }; - + } + ; function parseSql(text, type) { // reset values - cells = [] + cells = []; tableCell = null; rowCell = null; // load parser - const parser = new SqlSimpleParser(type); - - + const parser = new sqlsimpleparser_1.SqlSimpleParser(type); const models = parser .feed(text) .WithoutEnds() .WithEnds() .ToModel(); - - - foreignKeyList = models.ForeignKeyList; primaryKeyList = models.PrimaryKeyList; tableList = models.TableList; exportedTables = tableList.length; - //Create Table in UI CreateTableUI(type); - }; - /** - * return text quantifiers for dialect - * @returns json - */ - function GetColumnQuantifiers(type) { - let chars = { - Start: '"', - End: '"', - }; - if (type == "mysql") { - chars.Start = "`"; - chars.End = "`"; - } - else if (type == "sqlserver") { - chars.Start = "["; - chars.End = "]"; - } - return chars; - } - - /** - * extract row column attributes - * @param {*} label - * @param {*} columnQuantifiers - * @returns - */ - function getDbLabel(label, columnQuantifiers){ - // fix duplicate spaces and different space chars - label = label - .replace(/\s+/g, " ") - let firstSpaceIndex = label[0] == columnQuantifiers.Start && - label.indexOf(columnQuantifiers.End + " ") !== -1 - ? label.indexOf(columnQuantifiers.End + " ") - : label.indexOf(" "); - let attributeType = label.substring(firstSpaceIndex + 1).trim(); - let attributeName = label.substring(0, firstSpaceIndex + 1); - let attribute = { - attributeName, - attributeType - } - return attribute } - + ; function CreateTableUI(type) { - tableList.forEach(function(tableModel) { + tableList.forEach(function (tableModel) { //Define table size width - var maxNameLenght = 100 + tableModel.Name.length; - + const maxNameLenght = 100 + tableModel.Name.length; //Create Table - tableCell = new mxCell(tableModel.Name, new mxGeometry(dx, 0, maxNameLenght, 26), - 'swimlane;fontStyle=0;childLayout=stackLayout;horizontal=1;startSize=26;fillColor=default;horizontalStack=0;resizeParent=1;resizeLast=0;collapsible=1;marginBottom=0;swimlaneFillColor=default;align=center;'); + tableCell = new mxCell(tableModel.Name, new mxGeometry(dx, 0, maxNameLenght, 26), "swimlane;fontStyle=0;childLayout=stackLayout;horizontal=1;startSize=26;fillColor=default;horizontalStack=0;resizeParent=1;resizeLast=0;collapsible=1;marginBottom=0;swimlaneFillColor=default;align=center;"); tableCell.vertex = true; - //Resize row - var size = ui.editor.graph.getPreferredSizeForCell(rowCell); - if (size !== null) { - tableCell.geometry.width = size.width + maxNameLenght; + if (rowCell) { + const size = ui.editor.graph.getPreferredSizeForCell(rowCell); + if (size !== null) { + tableCell.geometry.width = size.width + maxNameLenght; + } } - //Add Table to cells cells.push(tableCell); - //Add properties - tableModel.Properties.forEach(function(propertyModel) { - + tableModel.Properties.forEach(function (propertyModel) { //Add row AddRow(propertyModel, tableModel.Name); }); - //Close table dx += tableCell.geometry.width + 40; tableCell = null; }); - if (cells.length > 0) { - var graph = ui.editor.graph; - var view = graph.view; - var bds = graph.getGraphBounds(); - + const graph = ui.editor.graph; + const view = graph.view; + const bds = graph.getGraphBounds(); // Computes unscaled, untranslated graph bounds - var x = Math.ceil(Math.max(0, bds.x / view.scale - view.translate.x) + 4 * graph.gridSize); - var y = Math.ceil(Math.max(0, (bds.y + bds.height) / view.scale - view.translate.y) + 4 * graph.gridSize); - + const x = Math.ceil(Math.max(0, bds.x / view.scale - view.translate.x) + 4 * graph.gridSize); + const y = Math.ceil(Math.max(0, (bds.y + bds.height) / view.scale - view.translate.y) + 4 * graph.gridSize); graph.setSelectionCells(graph.importCells(cells, x, y)); // add foreign key edges - var model = graph.getModel(); + const model = graph.getModel(); const columnQuantifiers = GetColumnQuantifiers(type); - var pt = graph.getFreeInsertPoint(); - foreignKeyList.forEach(function(fk){ - if(fk.IsDestination && fk.PrimaryKeyName && fk.ReferencesPropertyName && + const pt = graph.getFreeInsertPoint(); + foreignKeyList.forEach(function (fk) { + if (fk.IsDestination && fk.PrimaryKeyName && fk.ReferencesPropertyName && fk.PrimaryKeyTableName && fk.ReferencesTableName) { - var insertEdge = mxUtils.bind(this, function(targetCell, sourceCell, edge){ - var label = "" - var edgeStyle = "edgeStyle=entityRelationEdgeStyle;html=1;endArrow=ERzeroToMany;startArrow=ERzeroToOne;labelBackgroundColor=none;fontFamily=Verdana;fontSize=14;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=-0.018;entryY=0.608;entryDx=0;entryDy=0;entryPerimeter=0;" - var edgeCell = graph.insertEdge(null, null, label || '', (edge.invert) ? - sourceCell : targetCell, (edge.invert) ? targetCell : sourceCell, edgeStyle); + const insertEdge = mxUtils.bind(this, function (targetCell, sourceCell, edge) { + const label = ""; + const edgeStyle = "edgeStyle=entityRelationEdgeStyle;html=1;endArrow=ERzeroToMany;startArrow=ERzeroToOne;labelBackgroundColor=none;fontFamily=Verdana;fontSize=14;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=-0.018;entryY=0.608;entryDx=0;entryDy=0;entryPerimeter=0;"; + const edgeCell = graph.insertEdge(null, null, label || "", (edge.invert) ? + sourceCell : targetCell, (edge.invert) ? targetCell : sourceCell, edgeStyle); }); - let edge = { + const edge = { invert: true }; - var targetCell = null; - var sourceCell = null; + let targetCell = null; + let sourceCell = null; // locate edge source and target cells for (const key in model.cells) { - if(targetCell && sourceCell) + if (targetCell && sourceCell) break; if (Object.hasOwnProperty.call(model.cells, key)) { const mxcell = model.cells[key]; - if(mxcell.style && mxcell.style.trim().startsWith("swimlane;")){ - let entity = { + if (mxcell.style && mxcell.style.trim().startsWith("swimlane;")) { + const entity = { name: mxcell.value, attributes: [] - } - var isPrimaryTable = entity.name == fk.PrimaryKeyTableName; - var isForeignTable = entity.name == fk.ReferencesTableName; - if(isPrimaryTable || isForeignTable){ + }; + const isPrimaryTable = entity.name == fk.PrimaryKeyTableName; + const isForeignTable = entity.name == fk.ReferencesTableName; + if (isPrimaryTable || isForeignTable) { for (let c = 0; c < mxcell.children.length; c++) { - if(targetCell && sourceCell) + if (targetCell && sourceCell) break; const col = mxcell.children[c]; - if(col.mxObjectId.indexOf("mxCell") !== -1) { - if(col.style && col.style.trim().startsWith("shape=partialRectangle")){ - let attribute = getDbLabel(col.value, columnQuantifiers) - if(isPrimaryTable && attribute.attributeName == fk.PrimaryKeyName){ + if (col.mxObjectId.indexOf("mxCell") !== -1) { + if (col.style && col.style.trim().startsWith("shape=partialRectangle")) { + const attribute = getDbLabel(col.value, columnQuantifiers); + if (isPrimaryTable && attribute.attributeName == fk.PrimaryKeyName) { targetCell = col; break; - } else if(isForeignTable && attribute.attributeName == fk.ReferencesPropertyName){ + } + else if (isForeignTable && attribute.attributeName == fk.ReferencesPropertyName) { sourceCell = col; break; } @@ -1364,95 +1386,88 @@ Draw.loadPlugin(function(ui) { } } } - } } } - if(targetCell && sourceCell) + if (targetCell && sourceCell) insertEdge(targetCell, sourceCell, edge); } - }) + }); graph.scrollCellToVisible(graph.getSelectionCell()); } - wndFromSQL.setVisible(false); - }; - + } + ; mxUtils.br(divFromSQL); - - const resetBtnFromSQL = mxUtils.button(mxResources.get('reset'), function() { + const resetBtnFromSQL = mxUtils.button(mxResources.get("reset"), function () { sqlInputFromSQL.value = defaultReset; }); - - resetBtnFromSQL.style.marginTop = '8px'; - resetBtnFromSQL.style.marginRight = '4px'; - resetBtnFromSQL.style.padding = '4px'; + resetBtnFromSQL.style.marginTop = "8px"; + resetBtnFromSQL.style.marginRight = "4px"; + resetBtnFromSQL.style.padding = "4px"; divFromSQL.appendChild(resetBtnFromSQL); - - const btnFromSQL_mysql = mxUtils.button('Insert MySQL', function() { - parseSql(sqlInputFromSQL.value, 'mysql'); + const btnFromSQL_mysql = mxUtils.button("Insert MySQL", function () { + parseSql(sqlInputFromSQL.value, "mysql"); }); - - btnFromSQL_mysql.style.marginTop = '8px'; - btnFromSQL_mysql.style.padding = '4px'; + btnFromSQL_mysql.style.marginTop = "8px"; + btnFromSQL_mysql.style.padding = "4px"; divFromSQL.appendChild(btnFromSQL_mysql); - - const btnFromSQL_sqlserver = mxUtils.button('Insert SQL Server', function() { - parseSql(sqlInputFromSQL.value, 'sqlserver'); + const btnFromSQL_sqlserver = mxUtils.button("Insert SQL Server", function () { + parseSql(sqlInputFromSQL.value, "sqlserver"); }); - - btnFromSQL_sqlserver.style.marginTop = '8px'; - btnFromSQL_sqlserver.style.padding = '4px'; + btnFromSQL_sqlserver.style.marginTop = "8px"; + btnFromSQL_sqlserver.style.padding = "4px"; divFromSQL.appendChild(btnFromSQL_sqlserver); - - const btnFromSQL_postgres = mxUtils.button('Insert PostgreSQL', function() { - parseSql(sqlInputFromSQL.value, 'postgres'); + const btnFromSQL_postgres = mxUtils.button("Insert PostgreSQL", function () { + parseSql(sqlInputFromSQL.value, "postgres"); }); - - btnFromSQL_postgres.style.marginTop = '8px'; - btnFromSQL_postgres.style.padding = '4px'; + btnFromSQL_postgres.style.marginTop = "8px"; + btnFromSQL_postgres.style.padding = "4px"; divFromSQL.appendChild(btnFromSQL_postgres); - - const btnFromSQL_sqlite = mxUtils.button('Insert Sqlite', function() { - parseSql(sqlInputFromSQL.value, 'sqlite'); + const btnFromSQL_sqlite = mxUtils.button("Insert Sqlite", function () { + parseSql(sqlInputFromSQL.value, "sqlite"); }); - - btnFromSQL_sqlite.style.marginTop = '8px'; - btnFromSQL_sqlite.style.padding = '4px'; + btnFromSQL_sqlite.style.marginTop = "8px"; + btnFromSQL_sqlite.style.padding = "4px"; divFromSQL.appendChild(btnFromSQL_sqlite); - // Adds action - ui.actions.addAction('fromSql', function() { + ui.actions.addAction("fromSql", function () { wndFromSQL.setVisible(!wndFromSQL.isVisible()); - if (wndFromSQL.isVisible()) { sqlInputFromSQL.focus(); } }); // end import diagrams from sql text methods - // finalize menu buttons - var theMenu = ui.menus.get('insert'); - var oldMenu = theMenu.funct; - - theMenu.funct = function(menu, parent) { - oldMenu.apply(this, arguments); - ui.menus.addMenuItems(menu, ['fromSql'], parent); - }; - if(theMenuExportAs && theMenuExportAs.enabled) { + const theMenu = ui.menus.get("insert"); + if (theMenu) { + const oldMenu = theMenu.funct; + theMenu.funct = function (...args) { + const [menu, parent] = args; + oldMenu.apply(this, args); + ui.menus.addMenuItems(menu, ["fromSql"], parent); + }; + } + if (theMenuExportAs && theMenuExportAs.enabled) { var oldMenuExportAs = theMenuExportAs.funct; - - theMenuExportAs.funct = function(menu, parent) { - oldMenuExportAs.apply(this, arguments); - ui.menus.addMenuItems(menu, ['tosql'], parent); + theMenuExportAs.funct = function (...args) { + const [menu, parent] = args; + oldMenuExportAs.apply(this, args); + ui.menus.addMenuItems(menu, ["tosql"], parent); }; - } else { + } + else { // vscode file export sql menu - var menu = ui.menus.get('file'); - var oldMenuExportAs = menu.funct; - menu.funct = function(menu, parent) { - oldMenuExportAs.apply(this, arguments); - ui.menus.addMenuItems(menu, ['tosql'], parent); - }; + const menu = ui.menus.get("file"); + if (menu && menu.enabled) { + var oldMenuExportAs = menu.funct; + menu.funct = function (...args) { + const [menu, parent] = args; + oldMenuExportAs.apply(this, args); + ui.menus.addMenuItems(menu, ["tosql"], parent); + }; + } } -}); \ No newline at end of file +}); + +},{"@funktechno/little-mermaid-2-the-sql/lib/src/generate-sql-ddl":1,"@funktechno/sqlsimpleparser":3}]},{},[4]); diff --git a/src/sql.ts b/src/sql.ts index 088ec85..6f609bb 100644 --- a/src/sql.ts +++ b/src/sql.ts @@ -72,9 +72,10 @@ Draw.loadPlugin(function(ui) { * @returns */ function removeHtml(label: string){ - const div = document.createElement("div"); - divGenSQL.innerHTML = label; - const text = div.textContent || div.innerText || ""; + const tempDiv = document.createElement("div"); + tempDiv.innerHTML = label; + const text = tempDiv.textContent || tempDiv.innerText || ""; + tempDiv.remove(); return text; } /** @@ -84,16 +85,15 @@ Draw.loadPlugin(function(ui) { * @returns */ function getDbLabel(label: string, columnQuantifiers: ColumnQuantifiers): TableAttribute{ - label = removeHtml(label); + let result = removeHtml(label); // fix duplicate spaces and different space chars - label = label - .replace(/\s+/g, " "); - const firstSpaceIndex = label[0] == columnQuantifiers.Start && - label.indexOf(columnQuantifiers.End + " ") !== -1 - ? label.indexOf(columnQuantifiers.End + " ") - : label.indexOf(" "); - const attributeType = label.substring(firstSpaceIndex + 1).trim(); - const attributeName = RemoveNameQuantifiers(label.substring(0, firstSpaceIndex)); + result = result.toString().replace(/\s+/g, " "); + const firstSpaceIndex = result[0] == columnQuantifiers.Start && + result.indexOf(columnQuantifiers.End + " ") !== -1 + ? result.indexOf(columnQuantifiers.End + " ") + : result.indexOf(" "); + const attributeType = result.substring(firstSpaceIndex + 1).trim(); + const attributeName = RemoveNameQuantifiers(result.substring(0, firstSpaceIndex + 1)); const attribute = { attributeName, attributeType From 979454c430919a4ec03c84fc23ba55afdab2e6e7 Mon Sep 17 00:00:00 2001 From: lastlink Date: Sun, 23 Jun 2024 12:15:27 -0400 Subject: [PATCH 12/12] minor changes --- build/updateVersion.js | 4 +- dist/sql.js | 2414 ++++++++++++++++++++-------------------- src/sql.ts | 26 +- 3 files changed, 1228 insertions(+), 1216 deletions(-) diff --git a/build/updateVersion.js b/build/updateVersion.js index b6a96e6..9ab1715 100644 --- a/build/updateVersion.js +++ b/build/updateVersion.js @@ -13,7 +13,9 @@ console.log("Updating to version " + package.version); files.forEach((file) => { const filePath = `${directoryPath}/${file}`; let content = fs.readFileSync(filePath, "utf8"); - content = content.replace(oldText, package.version); + while(content.includes(oldText)) { + content = content.replace(oldText, package.version); + } // content = content.replace(new RegExp(oldText, "g"), newText); fs.writeFileSync(filePath, content, "utf8"); console.log(`Replaced text in ${file}`); diff --git a/dist/sql.js b/dist/sql.js index 9f8aebb..96063ed 100644 --- a/dist/sql.js +++ b/dist/sql.js @@ -1,188 +1,940 @@ +(function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i 0) { + attributeName = testFullNameMatches[0]; + attributeComment = attributeComment + .replace(`'${attributeName}'`, "") + .trim(); + } + } + if (attributeComment) + attributeComment = " " + attributeComment; + // check if contains full column name + statement += `\t${this.dbTypeEnds(attributeName)} ${columnType}${attributeComment}`; + if (attribute.attributeKeyType && + attribute.attributeKeyType == "PK") { + primaryKeys.push(attribute.attributeName); + } + } + else { + if (attribute.attributeKeyType && + attribute.attributeKeyType == "PK") { + primaryKeys.push(attribute.attributeName); } + statement += `\t${this.dbTypeEnds(attribute.attributeName)} ${columnType}`; } + // statement += i != entity.attributes.length - 1 ? ",\n" : "\n"; } - return statementGeneration.join(""); } - /** - * convert labels with start and end strings per database type - * @param label - * @returns - */ - dbTypeEnds(label) { - let char1 = '"'; - let char2 = '"'; - if (this.dbType == "mysql") { - char1 = "`"; - char2 = "`"; + if (primaryKeys.length > 0) { + statement += ",\n\tPRIMARY KEY("; + for (let i = 0; i < primaryKeys.length; i++) { + statement += (i == 0 ? "" : ",") + this.dbTypeEnds(primaryKeys[i]); } - else if (this.dbType == "sqlserver") { - char1 = "["; - char2 = "]"; + statement += ")"; + } + // foreign keys + const entityFKeys = this.relationships.filter((relation) => relation.entityB == entityKey); + if (entityFKeys.length > 0) { + for (let i = 0; i < entityFKeys.length; i++) { + const fk = entityFKeys[i]; + const fkRelTxt = fk.roleA; + // must match format "[..] to [..]" + const keySplit = "] to ["; + if (fkRelTxt.indexOf(keySplit) != -1 && + fkRelTxt[0] == "[" && + fkRelTxt[fkRelTxt.length - 1] == "]") { + const keys = fkRelTxt.substring(1, fkRelTxt.length - 1).split(keySplit); + // remove quotes + let fkCol = keys[1].replace(/[\'\"]/gim, ""); + if (fkCol.indexOf(".") != -1) { + fkCol = fkCol.split(".")[1]; + } + fkCol = fkCol.trim(); + let pkCol = keys[0].replace(/[\'\"]/gim, ""); + if (pkCol.indexOf(".") != -1) { + pkCol = pkCol.split(".")[1]; + } + pkCol = pkCol.trim(); + // FOREIGN KEY (`Artist Id`) REFERENCES `Artist`(`ArtistId`) + statement += `,\n\tFOREIGN KEY (${this.dbTypeEnds(fkCol)}) REFERENCES ${this.dbTypeEnds(fk.entityA)}(${this.dbTypeEnds(pkCol)})`; + } } - return `${char1}${label}${char2}`; } + if (attributesAdded != 0) { + statement += "\n"; + } + statement += ");\n\n"; + return statement; + } + /* database create table examples: + sqlite example: + CREATE TABLE [test_table] ( + "id" INTEGER NOT NULL, + "Field 2_2" TEXT, + "Artist Id" INTEGER, + PRIMARY KEY("id"), + FOREIGN KEY("Artist Id") REFERENCES "Artist"("ArtistId") ON DELETE NO ACTION ON UPDATE NO ACTION + ) + + mysql example: + CREATE TABLE `test_table` ( + `id` INTEGER NOT NULL, + `Field 2_2` TEXT, + `Artist Id` INTEGER, + PRIMARY KEY (`id`), + FOREIGN KEY (`Artist Id`) REFERENCES `Artist`(`ArtistId`) + ); + + postgres example: + CREATE TABLE "test_table" ( + "id" INTEGER NOT NULL, + "Field 2_2" TEXT, + "Artist Id" INTEGER, + PRIMARY KEY ("id"), + FOREIGN KEY ("Artist Id"") REFERENCES "Artist"("ArtistId") + ); + + sql server example: + CREATE TABLE [test_table] ( + [id] INTEGER NOT NULL, + [Field 2_2] TEXT, + [Artist Id] INTEGER, + PRIMARY KEY ([id]), + FOREIGN KEY ([Artist Id]) REFERENCES [Artist]([ArtistId]) + ); + */ + createColumn(tableName, columnName, dataType) { + return `ALTER TABLE ${tableName} ADD ${columnName} ${dataType}`; + } +} +exports.DbParser = DbParser; + +},{}],2:[function(require,module,exports){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +exports.CONSTRAINT_Foreign_Key = exports.CONSTRAINT_Primary_Key = exports.Foreign_Key = exports.Primary_Key = exports.CONSTRAINT = exports.AlterTable = exports.CreateTable = void 0; +exports.CreateTable = "create table"; +exports.AlterTable = "alter table "; +exports.CONSTRAINT = "constraint"; +exports.Primary_Key = "primary key"; +exports.Foreign_Key = "foreign key"; +exports.CONSTRAINT_Primary_Key = "constraint primary key"; +exports.CONSTRAINT_Foreign_Key = "constraint foreign key"; + +},{}],3:[function(require,module,exports){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +exports.SqlSimpleParser = void 0; +const contants_1 = require("./contants"); +/* +An attempt to fix sql parser for importing entity diagrams into draw io see +https://github.com/jgraph/drawio/issues/1178 +https://drawio-app.com/entity-relationship-diagrams-with-draw-io/ + +*/ +/** + * Main Parser class + */ +class SqlSimpleParser { + /** + * Parser constructor. + * Default dialect is 'sqlite'. + * Options: "mysql" | "sqlite" | "postgres" | "sqlserver" . + * + * @param dialect SQL dialect ('sqlite'). + */ + constructor(dialect = "sqlite") { + this.tableList = []; + /** + * Parsed statements. + */ + this.statements = []; + /** + * Remains of string feed, after last parsed statement. + */ + this.remains = ""; /** - * generate create table statement - * also includes primary keys and foreign keys - * @param entityKey - * @param entity - * @returns + * Whether preparser is currently escaped. */ - createTable(entityKey, entity) { - let statement = `CREATE TABLE ${this.dbTypeEnds(entityKey)} (`; - let primaryKeys = []; - let attributesAdded = 0; - for (let i = 0; i < entity.attributes.length; i++) { - const attribute = entity.attributes[i]; - if (attribute.attributeType && attribute.attributeName) { - statement += attributesAdded == 0 ? "\n" : ",\n"; - attributesAdded++; - // need to add parenthesis or commas - // BIGGEST difference from original is that column types don't have to be fixed - let columnType = attribute.attributeType.trim(); - if (attribute.attributeComment) { - let attributeName = attribute.attributeName; - let attributeComment = attribute.attributeComment.trim(); - if (attribute.attributeComment.indexOf("'") != -1) { - // extract - const testFullNameMatches = attribute.attributeComment.match(/(?<=((?<=[\s,.:;"']|^)["']))(?:(?=(\\?))\2.)*?(?=\1)/gmu); - if (testFullNameMatches && testFullNameMatches.length > 0) { - attributeName = testFullNameMatches[0]; - attributeComment = attributeComment - .replace(`'${attributeName}'`, "") + this.escaped = false; + /** + * Current quote char of preparser. + */ + this.quoted = ""; + this.exportedTables = 0; + this.SQLServer = "sqlserver"; + this.MODE_SQLSERVER = false; + this.foreignKeyList = []; + this.primaryKeyList = []; + this.dialect = dialect; + this.MODE_SQLSERVER = + this.dialect !== undefined && + this.dialect !== null && + this.dialect == this.SQLServer; + } + /** + * Feed chunk of string into parser. + * + * @param chunk Chunk of string to be parsed. + */ + feed(chunk) { + // + const removedComments = chunk + // remove database comments, multiline, --, and // + .replace(/\/\*[\s\S]*?\*\/|\/\/|--.*/g, "") + .replace(/IF NOT EXISTS/gi, "") + .trim(); + const cleanedLines = removedComments + .split("\n") + // remove empty lines + .filter((n) => n) + // remove multiple spaces + .map((n) => n.replace(/\s+/g, " ").trim()); + // combine lines that are in parenthesis + const lines = []; + let insertSameLine = false; + cleanedLines.forEach((n) => { + if (lines.length > 0) { + if ((n[0] == "(" && + lines[lines.length - 1].toLocaleLowerCase().indexOf(contants_1.CreateTable) == + -1) || + insertSameLine) { + if (lines.length > 0) { + insertSameLine = true; + lines[lines.length - 1] += ` ${n}`; + if (n[0] == ")") + insertSameLine = false; + } + } + else if (lines[lines.length - 1].match(/CONSTRAINT/gi) && + (n.match(/FOREIGN KEY/gi) && !n.match(/CONSTRAINT/gi))) { + lines[lines.length - 1] += ` ${n}`; + } + // add to previous line if current has references and previous has foreign key + else if (lines[lines.length - 1].match(/FOREIGN KEY/gi) && + (n.match(/REFERENCES/gi) && !n.match(/FOREIGN KEY/gi))) { + lines[lines.length - 1] += ` ${n}`; + } + else if (n.substring(0, 2).toUpperCase() == "ON") { + lines[lines.length - 1] += ` ${n}`; + } + else { + lines.push(n); + } + } + else { + lines.push(n); + } + }); + // dx = 0, + // tableCell = null, + // cells = [], + // exportedTables = 0, + // tableList = [], + // foreignKeyList = [], + // rowCell = null; + let currentTableModel = null; + //Parse SQL to objects + for (let i = 0; i < lines.length; i++) { + // rowCell = null; + const tmp = lines[i].trim(); + const propertyRow = tmp.toLowerCase().trim(); + if (propertyRow[0] == ")") { + // close table + if (currentTableModel) { + this.tableList.push(currentTableModel); + currentTableModel = null; + } + continue; + } + //Parse Table + if (propertyRow.indexOf(contants_1.CreateTable) != -1) { + //Parse row + let name = tmp + .replace(this.stringToRegex(`/${contants_1.CreateTable}/gi`), "") + .trim(); + //Parse Table Name + name = this.ParseTableName(name); + if (currentTableModel !== null) { + //Add table to the list + this.tableList.push(currentTableModel); + } + //Create Table + currentTableModel = this.CreateTable(name); + } + // Parse Properties + else if (tmp !== "(" && + currentTableModel != null && + propertyRow.indexOf(contants_1.AlterTable) == -1) { + //Parse the row + let name = tmp.substring(0, tmp.charAt(tmp.length - 1) === "," ? tmp.length - 1 : tmp.length); + //Attempt to get the Key Type + let propertyType = name.toLowerCase(); + // .substring(0, AlterTable.length).toLowerCase(); + //Add special constraints + if (this.MODE_SQLSERVER) { + if (propertyType.indexOf(contants_1.CONSTRAINT) !== -1 && + propertyType.indexOf(contants_1.Primary_Key) !== -1) { + propertyType = contants_1.CONSTRAINT_Primary_Key; + } + if (propertyType.indexOf(contants_1.CONSTRAINT) !== -1 && + propertyType.indexOf(contants_1.Foreign_Key) !== -1) { + propertyType = contants_1.CONSTRAINT_Foreign_Key; + } + } + //Verify if this is a property that doesn't have a relationship (One minute of silence for the property) + // TODO: make primary key check a regex match/ contains w/ () space or not + const normalProperty = !propertyType.match(/PRIMARY KEY\s?\(/gi) && + propertyType.indexOf(contants_1.Foreign_Key) == -1 && + propertyType.indexOf(contants_1.CONSTRAINT_Primary_Key) == -1 && + propertyType.indexOf(contants_1.CONSTRAINT_Foreign_Key) == -1; + const nameSkipCheck = name.toUpperCase().trim(); + //Parse properties that don't have relationships + if (normalProperty) { + if (name === "" || name === "" || name === ");") { + continue; + } + let ExtendedProperties = null; + if (this.MODE_SQLSERVER) { + if (nameSkipCheck.indexOf(" ASC") !== -1 || + nameSkipCheck.indexOf(" DESC") !== -1 || + nameSkipCheck.indexOf("EXEC ") !== -1 || + nameSkipCheck.indexOf("WITH ") !== -1 || + nameSkipCheck.indexOf(" ON") !== -1 || + nameSkipCheck.indexOf("ALTER ") !== -1 || + // comments already removed + nameSkipCheck.indexOf("/*") !== -1 || + nameSkipCheck.indexOf(" CONSTRAINT") !== -1 || + nameSkipCheck.indexOf("SET ") !== -1 || + nameSkipCheck.indexOf(" NONCLUSTERED") !== -1 || + // no spaces desired + nameSkipCheck.indexOf("GO") !== -1 || + nameSkipCheck.indexOf("REFERENCES ") !== -1) { + continue; + } + //Get delimiter of column name + //TODO: check for space? or end quantifier + const firstSpaceIndex = name[0] == "[" && name.indexOf("]" + " ") !== -1 + ? name.indexOf("]" + " ") + : name.indexOf(" "); + ExtendedProperties = name.substring(firstSpaceIndex + 1).trim(); + //Get full name + name = name.substring(0, firstSpaceIndex); + name = this.RemoveNameQuantifiers(name); + } + else { + const columnQuantifiers = this.GetColumnQuantifiers(); + //Get delimiter of column name + const firstSpaceIndex = name[0] == columnQuantifiers.Start && + name.indexOf(columnQuantifiers.End + " ") !== -1 + ? name.indexOf(columnQuantifiers.End + " ") + : name.indexOf(" "); + ExtendedProperties = name.substring(firstSpaceIndex + 1).trim(); + //Get full name + name = name.substring(0, firstSpaceIndex); + name = this.RemoveNameQuantifiers(name); + } + //Create Property + const propertyModel = this.CreateProperty(name, currentTableModel.Name, null, false, ExtendedProperties); + //Add Property to table + currentTableModel.Properties.push(propertyModel); + if (ExtendedProperties.toLocaleLowerCase().indexOf(contants_1.Primary_Key) > -1) { + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(name, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + } + } + else { + //Parse Primary Key + if (propertyType.indexOf(contants_1.Primary_Key) != -1 || + propertyType.indexOf(contants_1.CONSTRAINT_Primary_Key) != -1) { + if (!this.MODE_SQLSERVER) { + const primaryKey = name + .replace(/PRIMARY KEY\s?\(/gi, "") + .replace(")", ""); + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(primaryKey, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + } + else { + // let start = i + 2; + // let end = 0; + if (propertyRow.indexOf(contants_1.Primary_Key) !== -1 && + nameSkipCheck.indexOf("CLUSTERED") === -1) { + const primaryKey = name + .replace(/PRIMARY KEY\s?\(/gi, "") + .replace(")", ""); + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(primaryKey, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + } + else { + const startIndex = name.toLocaleLowerCase().indexOf("("); + const endIndex = name.indexOf(")") + 1; + const primaryKey = name + .substring(startIndex, endIndex) + .replace("(", "") + .replace(")", "") + .replace(/ASC/gi, "") .trim(); + const columnQuantifiers = this.GetColumnQuantifiers(); + //Get delimiter of column name + const firstSpaceIndex = primaryKey[0] == columnQuantifiers.Start && + primaryKey.indexOf(columnQuantifiers.End + " ") !== -1 + ? primaryKey.indexOf(columnQuantifiers.End + " ") + : primaryKey.indexOf(" "); + const primaryKeyRow = firstSpaceIndex == -1 + ? primaryKey + : primaryKey.substring(firstSpaceIndex + 1).trim(); + //Create Primary Key + const primaryKeyModel = this.CreatePrimaryKey(primaryKeyRow, currentTableModel.Name); + //Add Primary Key to List + this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); + /* + while (end === 0) { + let primaryKeyRow = lines[start].trim(); + + if (primaryKeyRow.indexOf(")") !== -1) { + end = 1; + break; + } + + start++; + + primaryKeyRow = primaryKeyRow.replace(/ASC/gi, ""); + + //Parse name + primaryKeyRow = this.ParseSQLServerName(primaryKeyRow, true); + + //Create Primary Key + let primaryKeyModel = this.CreatePrimaryKey( + primaryKeyRow, + currentTableModel.Name + ); + + //Add Primary Key to List + this.primaryKeyList.push(primaryKeyModel); + } + */ } } - if (attributeComment) - attributeComment = " " + attributeComment; - // check if contains full column name - statement += `\t${this.dbTypeEnds(attributeName)} ${columnType}${attributeComment}`; - if (attribute.attributeKeyType && - attribute.attributeKeyType == "PK") { - primaryKeys.push(attribute.attributeName); - } + } + } + //Parse Foreign Key + if (propertyType.indexOf(contants_1.Foreign_Key) != -1 || + propertyType.indexOf(contants_1.CONSTRAINT_Foreign_Key) != -1) { + if (!this.MODE_SQLSERVER || propertyRow.indexOf(contants_1.AlterTable) == -1) { + this.ParseMySQLForeignKey(name, currentTableModel); } else { - if (attribute.attributeKeyType && - attribute.attributeKeyType == "PK") { - primaryKeys.push(attribute.attributeName); + let completeRow = name; + if (nameSkipCheck.indexOf("REFERENCES") === -1) { + const referencesRow = lines[i + 1].trim(); + completeRow = + "ALTER TABLE [dbo].[" + + currentTableModel.Name + + "] WITH CHECK ADD" + + " " + + name + + " " + + referencesRow; } - statement += `\t${this.dbTypeEnds(attribute.attributeName)} ${columnType}`; + this.ParseSQLServerForeignKey(completeRow, currentTableModel); } } } - if (primaryKeys.length > 0) { - statement += ",\n\tPRIMARY KEY("; - for (let i = 0; i < primaryKeys.length; i++) { - const element = primaryKeys[i]; - statement += (i == 0 ? "" : ",") + this.dbTypeEnds(primaryKeys[i]); + else if (propertyRow.indexOf(contants_1.AlterTable) != -1) { + if (this.MODE_SQLSERVER) { + //Parse the row + const alterTableRow = tmp.substring(0, tmp.charAt(tmp.length - 1) === "," ? tmp.length - 1 : tmp.length); + const referencesRow = lines[i + 1].trim(); + const completeRow = alterTableRow + " " + referencesRow; + this.ParseSQLServerForeignKey(completeRow, currentTableModel); } - statement += ")"; } - // foreign keys - let entityFKeys = this.relationships.filter((relation) => relation.entityB == entityKey); - if (entityFKeys.length > 0) { - for (let i = 0; i < entityFKeys.length; i++) { - const fk = entityFKeys[i]; - const fkRelTxt = fk.roleA; - // must match format "[..] to [..]" - const keySplit = "] to ["; - if (fkRelTxt.indexOf(keySplit) != -1 && - fkRelTxt[0] == "[" && - fkRelTxt[fkRelTxt.length - 1] == "]") { - let keys = fkRelTxt.substring(1, fkRelTxt.length - 1).split(keySplit); - // remove quotes - let fkCol = keys[1].replace(/[\'\"]/gim, ""); - if (fkCol.indexOf(".") != -1) { - fkCol = fkCol.split(".")[1]; - } - fkCol = fkCol.trim(); - let pkCol = keys[0].replace(/[\'\"]/gim, ""); - if (pkCol.indexOf(".") != -1) { - pkCol = pkCol.split(".")[1]; + } + // parse fk and primary keys + if (this.primaryKeyList.length > 0) { + this.primaryKeyList.forEach((pk) => { + // find table index + const pkTableIndex = this.tableList.findIndex((t) => t.Name.toLocaleLowerCase() == + pk.PrimaryKeyTableName.toLocaleLowerCase()); + // find property index + if (pkTableIndex > -1) { + const propertyIndex = this.tableList[pkTableIndex].Properties.findIndex((p) => p.Name.toLocaleLowerCase() == + pk.PrimaryKeyName.toLocaleLowerCase()); + if (propertyIndex > -1) { + this.tableList[pkTableIndex].Properties[propertyIndex].IsPrimaryKey = true; + } + } + }); + } + if (this.foreignKeyList.length > 0) { + this.foreignKeyList.forEach((fk) => { + // find table index + const pkTableIndex = this.tableList.findIndex((t) => t.Name.toLocaleLowerCase() == + fk.ReferencesTableName.toLocaleLowerCase()); + // let fkTableIndex = this.tableList.findIndex( + // (t) => t.Name == fk.PrimaryKeyTableName + // ); + // find property index + if (pkTableIndex > -1) { + const propertyIndex = this.tableList[pkTableIndex].Properties.findIndex((p) => p.Name.toLocaleLowerCase() == + fk.PrimaryKeyName.toLocaleLowerCase()); + if (propertyIndex > -1) { + this.tableList[pkTableIndex].Properties[propertyIndex].ForeignKey.push(fk); + if (!fk.IsDestination) { + this.tableList[pkTableIndex].Properties[propertyIndex].IsForeignKey = true; } - pkCol = pkCol.trim(); - statement += `,\n\tFOREIGN KEY (${this.dbTypeEnds(fkCol)}) REFERENCES ${this.dbTypeEnds(fk.entityA)}(${this.dbTypeEnds(pkCol)})`; } } + // if (fkTableIndex > -1) { + // let propertyIndex = this.tableList[fkTableIndex].Properties.findIndex( + // (p) => p.Name == fk.PrimaryKeyName + // ); + // if (propertyIndex > -1) { + // this.tableList[fkTableIndex].Properties[propertyIndex].ForeignKey.push(fk) + // } + // } + }); + } + return this; + } + stringToRegex(str) { + // Main regex + const mainResult = str.match(/\/(.+)\/.*/); + const optionsResult = str.match(/\/.+\/(.*)/); + if (mainResult && optionsResult) { + const main = mainResult[1]; + // Regex options + const options = optionsResult[1]; + // Compiled regex + return new RegExp(main, options); + } + return new RegExp("//(.+)/.*/", "//.+/(.*)/"); + } + CreatePrimaryKey(primaryKeyName, primaryKeyTableName) { + const primaryKeyNames = this.RemoveNameQuantifiers(primaryKeyName) + .split(",") + .filter((n) => n) + // remove multiple spaces + .map((n) => n.replace(/\s+/g, " ").trim()); + const primaryKeys = []; + primaryKeyNames.forEach(name => { + const primaryKey = { + PrimaryKeyTableName: primaryKeyTableName, + PrimaryKeyName: name, + }; + primaryKeys.push(primaryKey); + }); + return primaryKeys; + } + CreateProperty(name, tableName, foreignKey, isPrimaryKey, columnProps) { + const isForeignKey = foreignKey !== undefined && foreignKey !== null; + const property = { + Name: name, + ColumnProperties: columnProps, + TableName: tableName, + ForeignKey: foreignKey || [], + IsForeignKey: isForeignKey, + IsPrimaryKey: isPrimaryKey, + }; + return property; + } + ParseMySQLForeignKey(name, currentTableModel) { + const referencesIndex = name.toLowerCase().indexOf("references"); + let foreignKeySQL = name.substring(0, referencesIndex); + foreignKeySQL = foreignKeySQL.substring(foreignKeySQL.toUpperCase().indexOf("FOREIGN KEY")); + let referencesSQL = name.substring(referencesIndex, name.length); + //Remove references syntax + referencesSQL = referencesSQL.replace(/REFERENCES /gi, ""); + //Get Table and Property Index + const referencedTableIndex = referencesSQL.indexOf("("); + const referencedPropertyIndex = referencesSQL.indexOf(")"); + //Get Referenced Table + const referencedTableName = referencesSQL.substring(0, referencedTableIndex); + //Get Referenced Key + const referencedPropertyName = referencesSQL.substring(referencedTableIndex + 1, referencedPropertyIndex); + //Get ForeignKey + const foreignKey = foreignKeySQL + .replace(/FOREIGN KEY\s?\(/gi, "") + .replace(")", "") + .replace(" ", ""); + //Create ForeignKey + const foreignKeyOriginModel = this.CreateForeignKey(foreignKey, currentTableModel.Name, referencedPropertyName, referencedTableName, true); + //Add ForeignKey Origin + this.foreignKeyList.push(foreignKeyOriginModel); + //Create ForeignKey + const foreignKeyDestinationModel = this.CreateForeignKey(referencedPropertyName, referencedTableName, foreignKey, currentTableModel.Name, false); + //Add ForeignKey Destination + this.foreignKeyList.push(foreignKeyDestinationModel); + } + ParseSQLServerForeignKey(name, currentTableModel) { + const referencesIndex = name.toLowerCase().indexOf("references"); + let foreignKeySQL = ""; + if (name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`) !== -1) { + foreignKeySQL = name + .substring(name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`), referencesIndex) + .replace(/FOREIGN KEY\(/gi, "") + .replace(")", ""); + } + else { + foreignKeySQL = name + .substring(name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`), referencesIndex) + .replace(/FOREIGN KEY\s?\(/gi, "") + .replace(")", ""); + } + let referencesSQL = name.substring(referencesIndex, name.length); + const nameSkipCheck = name.toUpperCase().trim(); + let alterTableName = name + .substring(0, nameSkipCheck.indexOf("WITH")) + .replace(/ALTER TABLE /gi, ""); + if (referencesIndex !== -1 && + alterTableName !== "" && + foreignKeySQL !== "" && + referencesSQL !== "") { + //Remove references syntax + referencesSQL = referencesSQL.replace(/REFERENCES /gi, ""); + //Get Table and Property Index + const referencedTableIndex = referencesSQL.indexOf("("); + const referencedPropertyIndex = referencesSQL.indexOf(")"); + //Get Referenced Table + let referencedTableName = referencesSQL.substring(0, referencedTableIndex); + //Parse Name + referencedTableName = this.ParseSQLServerName(referencedTableName); + //Get Referenced Key + let referencedPropertyName = referencesSQL.substring(referencedTableIndex + 1, referencedPropertyIndex); + //Parse Name + referencedPropertyName = this.ParseSQLServerName(referencedPropertyName); + //Get ForeignKey + let foreignKey = foreignKeySQL + .replace(/FOREIGN KEY\s?\(/gi, "") + .replace(")", ""); + //Parse Name + foreignKey = this.ParseSQLServerName(foreignKey); + //Parse Name + alterTableName = this.ParseSQLServerName(alterTableName); + //Create ForeignKey + const foreignKeyOriginModel = this.CreateForeignKey(foreignKey, alterTableName, referencedPropertyName, referencedTableName, true); + //Add ForeignKey Origin + this.foreignKeyList.push(foreignKeyOriginModel); + //Create ForeignKey + const foreignKeyDestinationModel = this.CreateForeignKey(referencedPropertyName, referencedTableName, foreignKey, alterTableName, false); + //Add ForeignKey Destination + this.foreignKeyList.push(foreignKeyDestinationModel); + } + } + CreateForeignKey(primaryKeyName, primaryKeyTableName, referencesPropertyName, referencesTableName, isDestination) { + const foreignKey = { + PrimaryKeyTableName: this.RemoveNameQuantifiers(primaryKeyTableName), + PrimaryKeyName: this.RemoveNameQuantifiers(primaryKeyName), + ReferencesPropertyName: this.RemoveNameQuantifiers(referencesPropertyName), + ReferencesTableName: this.RemoveNameQuantifiers(referencesTableName), + IsDestination: isDestination !== undefined && isDestination !== null + ? isDestination + : false, + }; + return foreignKey; + } + RemoveNameQuantifiers(name) { + return name.replace(/\[|\]|\(|\"|\'|\`/g, "").trim(); + } + ParseTableName(name) { + if (name.charAt(name.length - 1) === "(") { + name = this.RemoveNameQuantifiers(name); + // if (!this.MODE_SQLSERVER) { + // name = name.substring(0, name.lastIndexOf(" ")); + // } else { + // name = this.ParseSQLServerName(name); + // } + } + return name; + } + ParseSQLServerName(name, property) { + name = name.replace("[dbo].[", ""); + name = name.replace("](", ""); + name = name.replace("].[", "."); + name = name.replace("[", ""); + if (property == undefined || property == null) { + name = name.replace(" [", ""); + name = name.replace("] ", ""); + } + else { + if (name.indexOf("]") !== -1) { + name = name.substring(0, name.indexOf("]")); } - if (attributesAdded != 0) { - statement += "\n"; - } - statement += `);\n\n`; - return statement; } + if (name.lastIndexOf("]") === name.length - 1) { + name = name.substring(0, name.length - 1); + } + if (name.lastIndexOf(")") === name.length - 1) { + name = name.substring(0, name.length - 1); + } + if (name.lastIndexOf("(") === name.length - 1) { + name = name.substring(0, name.length - 1); + } + name = name.replace(" ", ""); + return name; } - + CreateTable(name) { + const table = { + Name: name, + Properties: [], + }; + //Count exported tables + this.exportedTables++; + return table; + } + /** + * Checks whether character is a quotation character. + * + * @param char Character to be evaluated. + */ + static isQuoteChar(char) { + return char === "\"" || char === "'" || char === "`"; + } + /** + * convert labels with start and end strings per database type + * @param label + * @returns + */ + dbTypeEnds(label) { + let char1 = "\""; + let char2 = "\""; + if (this.dialect == "mysql") { + char1 = "`"; + char2 = "`"; + } + else if (this.dialect == "sqlserver") { + char1 = "["; + char2 = "]"; + } + return `${char1}${label}${char2}`; + } + WithEnds() { + this.tableList = this.tableList.map((table) => { + table.Name = this.dbTypeEnds(table.Name); + table.Properties = table.Properties.map((property) => { + property.Name = this.dbTypeEnds(property.Name); + property.TableName = this.dbTypeEnds(property.TableName); + return property; + }); + return table; + }); + this.primaryKeyList = this.primaryKeyList.map((primaryKey) => { + primaryKey.PrimaryKeyName = this.dbTypeEnds(primaryKey.PrimaryKeyName); + primaryKey.PrimaryKeyTableName = this.dbTypeEnds(primaryKey.PrimaryKeyTableName); + return primaryKey; + }); + this.foreignKeyList = this.foreignKeyList.map((foreignKey) => { + foreignKey.PrimaryKeyName = this.dbTypeEnds(foreignKey.PrimaryKeyName); + foreignKey.ReferencesPropertyName = this.dbTypeEnds(foreignKey.ReferencesPropertyName); + foreignKey.PrimaryKeyTableName = this.dbTypeEnds(foreignKey.PrimaryKeyTableName); + foreignKey.ReferencesTableName = this.dbTypeEnds(foreignKey.ReferencesTableName); + return foreignKey; + }); + return this; + } + WithoutEnds() { + this.tableList.map((table) => { + table.Name = this.RemoveNameQuantifiers(table.Name); + table.Properties = table.Properties.map((property) => { + property.Name = this.RemoveNameQuantifiers(property.Name); + property.TableName = this.RemoveNameQuantifiers(property.TableName); + return property; + }); + return table; + }); + this.primaryKeyList = this.primaryKeyList.map((primaryKey) => { + primaryKey.PrimaryKeyName = this.RemoveNameQuantifiers(primaryKey.PrimaryKeyName); + primaryKey.PrimaryKeyTableName = this.RemoveNameQuantifiers(primaryKey.PrimaryKeyTableName); + return primaryKey; + }); + this.foreignKeyList = this.foreignKeyList.map((foreignKey) => { + foreignKey.PrimaryKeyName = this.RemoveNameQuantifiers(foreignKey.PrimaryKeyName); + foreignKey.ReferencesPropertyName = this.RemoveNameQuantifiers(foreignKey.ReferencesPropertyName); + foreignKey.PrimaryKeyTableName = this.RemoveNameQuantifiers(foreignKey.PrimaryKeyTableName); + foreignKey.ReferencesTableName = this.RemoveNameQuantifiers(foreignKey.ReferencesTableName); + return foreignKey; + }); + return this; + } + /** + * return text quantifiers for dialect + * @returns json + */ + GetColumnQuantifiers() { + const chars = { + Start: "\"", + End: "\"", + }; + if (this.dialect == "mysql") { + chars.Start = "`"; + chars.End = "`"; + } + else if (this.dialect == "sqlserver") { + chars.Start = "["; + chars.End = "]"; + } + return chars; + } + ToTableList() { + return this.tableList; + } + ToPrimaryKeyList() { + return this.primaryKeyList; + } + ToForeignKeyList() { + return this.foreignKeyList; + } + ResetModel() { + this.tableList = []; + this.primaryKeyList = []; + this.foreignKeyList = []; + return this; + } + /** + * return full sql model + * @returns + */ + ToModel() { + return { + TableList: this.tableList, + Dialect: this.dialect, + ForeignKeyList: this.foreignKeyList, + PrimaryKeyList: this.primaryKeyList, + }; + } +} +exports.SqlSimpleParser = SqlSimpleParser; + +},{"./contants":2}],4:[function(require,module,exports){ +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +const generate_sql_ddl_1 = require("@funktechno/little-mermaid-2-the-sql/lib/src/generate-sql-ddl"); +const sqlsimpleparser_1 = require("@funktechno/sqlsimpleparser"); +/** + * SQL Tools Plugin for importing diagrams from SQL DDL and exporting to SQL. + * Version: 0.0.1 + */ +Draw.loadPlugin(function (ui) { + // export sql methods + const pluginVersion = "0.0.1"; //Create Base div - const divGenSQL = document.createElement('div'); - divGenSQL.style.userSelect = 'none'; - divGenSQL.style.overflow = 'hidden'; - divGenSQL.style.padding = '10px'; - divGenSQL.style.height = '100%'; - - const sqlInputGenSQL = document.createElement('textarea'); - sqlInputGenSQL.style.height = '200px'; - sqlInputGenSQL.style.width = '100%'; - sqlInputGenSQL.value = '-- click a database type button' + const divGenSQL = document.createElement("div"); + divGenSQL.style.userSelect = "none"; + divGenSQL.style.overflow = "hidden"; + divGenSQL.style.padding = "10px"; + divGenSQL.style.height = "100%"; + const sqlInputGenSQL = document.createElement("textarea"); + sqlInputGenSQL.style.height = "200px"; + sqlInputGenSQL.style.width = "100%"; + const sqlExportDefault = "-- click a database type button"; + sqlInputGenSQL.value = sqlExportDefault; mxUtils.br(divGenSQL); divGenSQL.appendChild(sqlInputGenSQL); - var theMenuExportAs = ui.menus.get('exportAs'); - let buttonLabel = 'tosql=To SQL' + const theMenuExportAs = ui.menus.get("exportAs"); + let buttonLabel = "tosql=To SQL"; // vscode extension support - if(!(theMenuExportAs && theMenuExportAs.enabled)) { - buttonLabel = 'tosql=Export As SQL' + if (!(theMenuExportAs && theMenuExportAs.enabled)) { + buttonLabel = "tosql=Export As SQL"; } // Extends Extras menu mxResources.parse(buttonLabel); - - const wndGenSQL = new mxWindow(mxResources.get('tosql'), divGenSQL, document.body.offsetWidth - 480, 140, - 320, 320, true, true); + const wndGenSQL = new mxWindow(mxResources.get("tosql"), divGenSQL, document.body.offsetWidth - 480, 140, 320, 320, true, true); wndGenSQL.destroyOnClose = false; wndGenSQL.setMaximizable(false); wndGenSQL.setResizable(false); wndGenSQL.setClosable(true); - /** * return text quantifiers for dialect * @returns json */ function GetColumnQuantifiers(type) { - let chars = { - Start: '"', - End: '"', + const chars = { + Start: "\"", + End: "\"", }; if (type == "mysql") { chars.Start = "`"; @@ -196,1167 +948,435 @@ Draw.loadPlugin(function(ui) { } /** * sometimes rows have spans or styles, an attempt to remove them - * @param {*} label - * @returns + * @param {*} label + * @returns */ - function removeHtml(label){ - var div = document.createElement("div"); - divGenSQL.innerHTML = label; - var text = div.textContent || div.innerText || ""; + function removeHtml(label) { + const tempDiv = document.createElement("div"); + tempDiv.innerHTML = label; + const text = tempDiv.textContent || tempDiv.innerText || ""; + tempDiv.remove(); return text; } /** * extract row column attributes - * @param {*} label - * @param {*} columnQuantifiers - * @returns + * @param {*} label + * @param {*} columnQuantifiers + * @returns */ - function getDbLabel(label, columnQuantifiers){ - label = removeHtml(label) + function getDbLabel(label, columnQuantifiers) { + label = removeHtml(label); // fix duplicate spaces and different space chars label = label - .replace(/\s+/g, " ") - let firstSpaceIndex = label[0] == columnQuantifiers.Start && + .replace(/\s+/g, " "); + const firstSpaceIndex = label[0] == columnQuantifiers.Start && label.indexOf(columnQuantifiers.End + " ") !== -1 - ? label.indexOf(columnQuantifiers.End + " ") - : label.indexOf(" "); - let attributeType = label.substring(firstSpaceIndex + 1).trim(); - let attributeName = RemoveNameQuantifiers(label.substring(0, firstSpaceIndex)); - let attribute = { + ? label.indexOf(columnQuantifiers.End + " ") + : label.indexOf(" "); + const attributeType = label.substring(firstSpaceIndex + 1).trim(); + const attributeName = RemoveNameQuantifiers(label.substring(0, firstSpaceIndex + 1)); + const attribute = { attributeName, attributeType - } - return attribute + }; + return attribute; } function RemoveNameQuantifiers(name) { return name.replace(/\[|\]|\(|\"|\'|\`/g, "").trim(); } - - function getMermaidDiagramDb(type){ - var model = ui.editor.graph.getModel() + function getMermaidDiagramDb(type) { + const model = ui.editor.graph.getModel(); // same models from mermaid for diagram relationships // only difference is entities is an array rather than object to allow duplicate tables - let entities = {} - let relationships = [] + const entities = {}; + const relationships = []; // build models for (const key in model.cells) { if (Object.hasOwnProperty.call(model.cells, key)) { const mxcell = model.cells[key]; - if(mxcell.mxObjectId.indexOf("mxCell") !== -1) { - if(mxcell.style && mxcell.style.trim().startsWith("swimlane;")){ - let entity = { + if (mxcell.mxObjectId.indexOf("mxCell") !== -1) { + if (mxcell.style && mxcell.style.trim().startsWith("swimlane;")) { + const entity = { name: RemoveNameQuantifiers(mxcell.value), - attributes: [] - } + attributes: [], + }; for (let c = 0; c < mxcell.children.length; c++) { const col = mxcell.children[c]; - if(col.mxObjectId.indexOf("mxCell") !== -1) { - if(col.style && col.style.trim().startsWith("shape=partialRectangle")){ + if (col.mxObjectId.indexOf("mxCell") !== -1) { + if (col.style && col.style.trim().startsWith("shape=partialRectangle")) { const columnQuantifiers = GetColumnQuantifiers(type); //Get delimiter of column name //Get full name - let attribute = getDbLabel(col.value, columnQuantifiers) - var attributeKeyType = col.children.find(x=> ["FK","PK"].findIndex(k => k== x.value.toUpperCase()) !== -1 || - x.value.toUpperCase().indexOf("PK,")!=-1) - if(attributeKeyType){ - attribute.attributeKeyType = attributeKeyType.value - if(attribute.attributeKeyType != "PK" && attribute.attributeKeyType.indexOf("PK") != -1){ - attribute.attributeKeyType = "PK" + const attribute = getDbLabel(col.value, columnQuantifiers); + const attributeKeyType = col.children.find(x => ["FK", "PK"].findIndex(k => k == x.value.toUpperCase()) !== -1 || + x.value.toUpperCase().indexOf("PK,") != -1); + if (attributeKeyType) { + attribute.attributeKeyType = attributeKeyType.value; + if (attribute.attributeKeyType != "PK" && attribute.attributeKeyType.indexOf("PK") != -1) { + attribute.attributeKeyType = "PK"; } } - entity.attributes.push(attribute) - if(col.edges && col.edges.length){ + entity.attributes.push(attribute); + if (col.edges && col.edges.length) { // check for edges foreign keys for (let e = 0; e < col.edges.length; e++) { const edge = col.edges[e]; - if(edge.mxObjectId.indexOf("mxCell") !== -1) { - if(edge.style && edge.style.indexOf("endArrow=") != -1 && edge.source && - edge.source.value && edge.target && edge.target.value){ - // need to check if end is open or certain value to determin relationship type - // extract endArrow txt - // check if both match and contain many or open - // if both match and are many then create a new table - let endCheck = "endArrow=" - let endArr = edge.style.indexOf(endCheck) != -1 ? + if (edge.mxObjectId.indexOf("mxCell") !== -1) { + if (edge.style && edge.style.indexOf("endArrow=") != -1 && edge.source && + edge.source.value && edge.target && edge.target.value) { + // need to check if end is open or certain value to determin relationship type + // extract endArrow txt + // check if both match and contain many or open + // if both match and are many then create a new table + const endCheck = "endArrow="; + const endArr = edge.style.indexOf(endCheck) != -1 ? edge.style.substring(edge.style.indexOf(endCheck) + endCheck.length, edge.style.substring(edge.style.indexOf(endCheck) + endCheck.length).indexOf(";") + edge.style.indexOf(endCheck) + endCheck.length) - : "" - let startCheck = "startArrow=" - let startArr = edge.style.indexOf(startCheck) != -1 ? + : ""; + const startCheck = "startArrow="; + const startArr = edge.style.indexOf(startCheck) != -1 ? edge.style.substring(edge.style.indexOf(startCheck) + startCheck.length, edge.style.substring(edge.style.indexOf(startCheck) + startCheck.length).indexOf(";") + edge.style.indexOf(startCheck) + startCheck.length) - : "" - - var manyCheck = ["open","many"] - var sourceIsPrimary = endArr && manyCheck - .findIndex(x => endArr.toLocaleLowerCase().indexOf(x)!=-1) != -1; - var targetIsPrimary = startArr && manyCheck - .findIndex(x => startArr.toLocaleLowerCase().indexOf(x)!=-1) != -1; - // has to be one to many and not one to one - if((targetIsPrimary || sourceIsPrimary) && - !(targetIsPrimary && sourceIsPrimary) - ){ - var sourceId = edge.source.value; - sourceAttr = getDbLabel(sourceId, columnQuantifiers); - sourceId = sourceAttr.attributeName - var sourceEntity = RemoveNameQuantifiers(edge.source.parent.value); - var targetId = edge.target.value; - targetAttr = getDbLabel(targetId, columnQuantifiers); - targetId = targetAttr.attributeName - var targetEntity = RemoveNameQuantifiers(edge.target.parent.value); - // entityA primary - // entityB foreign - let relationship = { - entityA: sourceIsPrimary ? sourceEntity : targetEntity, - entityB: sourceIsPrimary ? targetEntity : sourceEntity, - // based off of styles? - relSpec: { - cardA: 'ZERO_OR_MORE', - cardB: 'ONLY_ONE', - relType: "IDENTIFYING" - }, - roleA: sourceIsPrimary ? - `[${sourceEntity}.${sourceId}] to [${targetEntity}.${targetId}]` : - `[${targetEntity}.${targetId}] to [${sourceEntity}.${sourceId}]` - } - // check that is doesn't already exist - var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA) - if(exists ==-1){ - relationships.push(relationship) - } - } else if(targetIsPrimary && sourceIsPrimary){ - // add a new many to many table - var sourceId = edge.source.value; - sourceAttr = getDbLabel(sourceId, columnQuantifiers); - sourceAttr.attributeKeyType = "PK" - sourceId = sourceAttr.attributeName - var sourceEntity = RemoveNameQuantifiers(edge.source.parent.value); - var targetId = edge.target.value; - targetAttr = getDbLabel(targetId, columnQuantifiers); - targetAttr.attributeKeyType = "PK" - targetId = targetAttr.attributeName - var targetEntity = RemoveNameQuantifiers(edge.target.parent.value); - let compositeEntity = { - name: RemoveNameQuantifiers(sourceEntity) + "_" + RemoveNameQuantifiers(targetEntity), - attributes: [sourceAttr, targetAttr] - } - // add composite entity - if(entities[compositeEntity.name]){ - // DON'T add duplicate composite tables - } else { - entities[compositeEntity.name] = compositeEntity - } - // entityA primary - // entityB foreign - let relationship = { - entityA: sourceEntity, - entityB: compositeEntity.name, - // based off of styles? - relSpec: { - cardA: 'ZERO_OR_MORE', - cardB: 'ONLY_ONE', - relType: "IDENTIFYING" - }, - roleA: `[${sourceEntity}.${sourceId}] to [${compositeEntity.name}.${sourceId}]` - } - // check that is doesn't already exist - var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA) - if(exists ==-1){ - relationships.push(relationship) - } - let relationship2 = { - entityA: targetEntity, - entityB: compositeEntity.name, - // based off of styles? - relSpec: { - cardA: 'ZERO_OR_MORE', - cardB: 'ONLY_ONE', - relType: "IDENTIFYING" - }, - roleA: `[${targetEntity}.${targetId}] to [${compositeEntity.name}.${targetId}]` - } - // check that is doesn't already exist - exists = relationships.findIndex(r => r.entityA == relationship2.entityA && r.entityB == relationship2.entityB && r.roleA == relationship2.roleA) - if(exists ==-1){ - relationships.push(relationship2) - } + : ""; + const manyCheck = ["open", "many"]; + const sourceIsPrimary = endArr && manyCheck + .findIndex(x => endArr.toLocaleLowerCase().indexOf(x) != -1) != -1; + const targetIsPrimary = startArr && manyCheck + .findIndex(x => startArr.toLocaleLowerCase().indexOf(x) != -1) != -1; + // has to be one to many and not one to one + if ((targetIsPrimary || sourceIsPrimary) && + !(targetIsPrimary && sourceIsPrimary)) { + var sourceId = edge.source.value; + var sourceAttr = getDbLabel(sourceId, columnQuantifiers); + sourceId = sourceAttr.attributeName; + var sourceEntity = RemoveNameQuantifiers(edge.source.parent.value); + var targetId = edge.target.value; + var targetAttr = getDbLabel(targetId, columnQuantifiers); + targetId = targetAttr.attributeName; + var targetEntity = RemoveNameQuantifiers(edge.target.parent.value); + // entityA primary + // entityB foreign + const relationship = { + entityA: sourceIsPrimary ? sourceEntity : targetEntity, + entityB: sourceIsPrimary ? targetEntity : sourceEntity, + // based off of styles? + relSpec: { + cardA: "ZERO_OR_MORE", + cardB: "ONLY_ONE", + relType: "IDENTIFYING" + }, + roleA: sourceIsPrimary ? + `[${sourceEntity}.${sourceId}] to [${targetEntity}.${targetId}]` : + `[${targetEntity}.${targetId}] to [${sourceEntity}.${sourceId}]` + }; + // check that is doesn't already exist + var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA); + if (exists == -1) { + relationships.push(relationship); } - + } + else if (targetIsPrimary && sourceIsPrimary) { + // add a new many to many table + var sourceId = edge.source.value; + sourceAttr = getDbLabel(sourceId, columnQuantifiers); + sourceAttr.attributeKeyType = "PK"; + sourceId = sourceAttr.attributeName; + var sourceEntity = RemoveNameQuantifiers(edge.source.parent.value); + var targetId = edge.target.value; + targetAttr = getDbLabel(targetId, columnQuantifiers); + targetAttr.attributeKeyType = "PK"; + targetId = targetAttr.attributeName; + var targetEntity = RemoveNameQuantifiers(edge.target.parent.value); + const compositeEntity = { + name: RemoveNameQuantifiers(sourceEntity) + "_" + RemoveNameQuantifiers(targetEntity), + attributes: [sourceAttr, targetAttr] + }; + // add composite entity + if (entities[compositeEntity.name]) { + // DON'T add duplicate composite tables + } + else { + entities[compositeEntity.name] = compositeEntity; + } + // entityA primary + // entityB foreign + const relationship = { + entityA: sourceEntity, + entityB: compositeEntity.name, + // based off of styles? + relSpec: { + cardA: "ZERO_OR_MORE", + cardB: "ONLY_ONE", + relType: "IDENTIFYING" + }, + roleA: `[${sourceEntity}.${sourceId}] to [${compositeEntity.name}.${sourceId}]` + }; + // check that is doesn't already exist + var exists = relationships.findIndex(r => r.entityA == relationship.entityA && r.entityB == relationship.entityB && r.roleA == relationship.roleA); + if (exists == -1) { + relationships.push(relationship); + } + const relationship2 = { + entityA: targetEntity, + entityB: compositeEntity.name, + // based off of styles? + relSpec: { + cardA: "ZERO_OR_MORE", + cardB: "ONLY_ONE", + relType: "IDENTIFYING" + }, + roleA: `[${targetEntity}.${targetId}] to [${compositeEntity.name}.${targetId}]` + }; + // check that is doesn't already exist + exists = relationships.findIndex(r => r.entityA == relationship2.entityA && r.entityB == relationship2.entityB && r.roleA == relationship2.roleA); + if (exists == -1) { + relationships.push(relationship2); + } + } } } - } } } } } // allows for duplicates if another table has the same name - if(entities[entity.name]){ - var count = 2; - while(entities[entity.name + count.toString()]){ + if (entities[entity.name]) { + let count = 2; + while (entities[entity.name + count.toString()]) { count++; } - entities[entity.name + count.toString()] = entity - } else { - entities[entity.name] = entity + entities[entity.name + count.toString()] = entity; + } + else { + entities[entity.name] = entity; } } - } } } - - class DbDefinition{ - constructor(entities, relationships){ + class DatabaseModel { + constructor(entities, relationships) { this.entities = entities; this.relationships = relationships; } - - getEntities(){ - return this.entities + getEntities() { + return this.entities; } - - getRelationships(){ - return this.relationships + getRelationships() { + return this.relationships; } } - - var db = new DbDefinition(entities, relationships); - + const db = new DatabaseModel(entities, relationships); return db; } - function generateSql(type) { - // get diagram model - var db = getMermaidDiagramDb(type); + const db = getMermaidDiagramDb(type); // load parser - var parser = new DbParser(type, db) + const parser = new generate_sql_ddl_1.DbParser(type, db); // generate sql - var sql = parser.getSQLDataDefinition() - sql = `/*\n\tGenerated in drawio\n\tDatabase: ${type}\n*/\n\n` + sql + let sql = parser.getSQLDataDefinition(); + sql = `/*\n\tGenerated in drawio\n\tDatabase: ${type}\n\tPlugin: sql\n\tVersion: ${pluginVersion}\n*/\n\n` + sql; sql = sql.trim(); // update sql value in text area sqlInputGenSQL.value = sql; // TODO: use selection as well? - var modelSelected = ui.editor.graph.getSelectionModel() - }; - + const modelSelected = ui.editor.graph.getSelectionModel(); + } + ; mxUtils.br(divGenSQL); - - const resetBtnGenSQL = mxUtils.button(mxResources.get('reset'), function() { - sqlInputGenSQL.value = ''; + const resetBtnGenSQL = mxUtils.button(mxResources.get("reset"), function () { + sqlInputGenSQL.value = sqlExportDefault; }); - - resetBtnGenSQL.style.marginTop = '8px'; - resetBtnGenSQL.style.marginRight = '4px'; - resetBtnGenSQL.style.padding = '4px'; + resetBtnGenSQL.style.marginTop = "8px"; + resetBtnGenSQL.style.marginRight = "4px"; + resetBtnGenSQL.style.padding = "4px"; divGenSQL.appendChild(resetBtnGenSQL); - - const btnGenSQL_mysql = mxUtils.button('MySQL', function() { - generateSql('mysql'); + const btnGenSQL_mysql = mxUtils.button("MySQL", function () { + generateSql("mysql"); }); - - btnGenSQL_mysql.style.marginTop = '8px'; - btnGenSQL_mysql.style.padding = '4px'; + btnGenSQL_mysql.style.marginTop = "8px"; + btnGenSQL_mysql.style.padding = "4px"; divGenSQL.appendChild(btnGenSQL_mysql); - - const btnGenSQL_sqlserver = mxUtils.button('SQL Server', function() { - generateSql('sqlserver'); + const btnGenSQL_sqlserver = mxUtils.button("SQL Server", function () { + generateSql("sqlserver"); }); - - btnGenSQL_sqlserver.style.marginTop = '8px'; - btnGenSQL_sqlserver.style.padding = '4px'; + btnGenSQL_sqlserver.style.marginTop = "8px"; + btnGenSQL_sqlserver.style.padding = "4px"; divGenSQL.appendChild(btnGenSQL_sqlserver); - - const btnGenSQL_postgres = mxUtils.button('PostgreSQL', function() { - generateSql('postgres'); + const btnGenSQL_postgres = mxUtils.button("PostgreSQL", function () { + generateSql("postgres"); }); - - btnGenSQL_postgres.style.marginTop = '8px'; - btnGenSQL_postgres.style.padding = '4px'; + btnGenSQL_postgres.style.marginTop = "8px"; + btnGenSQL_postgres.style.padding = "4px"; divGenSQL.appendChild(btnGenSQL_postgres); - - const btnGenSQL_sqlite = mxUtils.button('Sqlite', function() { - generateSql('sqlite'); + const btnGenSQL_sqlite = mxUtils.button("Sqlite", function () { + generateSql("sqlite"); }); - - btnGenSQL_sqlite.style.marginTop = '8px'; - btnGenSQL_sqlite.style.padding = '4px'; + btnGenSQL_sqlite.style.marginTop = "8px"; + btnGenSQL_sqlite.style.padding = "4px"; divGenSQL.appendChild(btnGenSQL_sqlite); - // Adds action - ui.actions.addAction('tosql', function() { + ui.actions.addAction("tosql", function () { wndGenSQL.setVisible(!wndGenSQL.isVisible()); - if (wndGenSQL.isVisible()) { sqlInputGenSQL.focus(); } }); // end export sql methods - // import diagrams from sql text methods - - const contants_1 ={} - contants_1.CONSTRAINT_Foreign_Key = contants_1.CONSTRAINT_Primary_Key = contants_1.Foreign_Key = contants_1.Primary_Key = contants_1.CONSTRAINT = contants_1.AlterTable = contants_1.CreateTable = void 0; - contants_1.CreateTable = "create table"; - contants_1.AlterTable = "alter table "; - contants_1.CONSTRAINT = "constraint"; - contants_1.Primary_Key = "primary key"; - contants_1.Foreign_Key = "foreign key"; - contants_1.CONSTRAINT_Primary_Key = "constraint primary key"; - contants_1.CONSTRAINT_Foreign_Key = "constraint foreign key"; - - /** - * Main Parser class - */ - class SqlSimpleParser { - /** - * Parser constructor. - * Default dialect is 'sqlite'. - * Options: "mysql" | "sqlite" | "postgres" | "sqlserver" . - * - * @param dialect SQL dialect ('sqlite'). - */ - constructor(dialect = "sqlite") { - this.tableList = []; - /** - * Parsed statements. - */ - this.statements = []; - /** - * Remains of string feed, after last parsed statement. - */ - this.remains = ""; - /** - * Whether preparser is currently escaped. - */ - this.escaped = false; - /** - * Current quote char of preparser. - */ - this.quoted = ""; - this.exportedTables = 0; - this.SQLServer = "sqlserver"; - this.MODE_SQLSERVER = false; - this.foreignKeyList = []; - this.primaryKeyList = []; - this.dialect = dialect; - this.MODE_SQLSERVER = - this.dialect !== undefined && - this.dialect !== null && - this.dialect == this.SQLServer; - } - /** - * Feed chunk of string into parser. - * - * @param chunk Chunk of string to be parsed. - */ - feed(chunk) { - // - const removedComments = chunk - // remove database comments, multiline, --, and // - .replace(/\/\*[\s\S]*?\*\/|\/\/|--.*/g, "") - .replace(/IF NOT EXISTS/gi, "") - .trim(); - const cleanedLines = removedComments - .split("\n") - // remove empty lines - .filter((n) => n) - // remove multiple spaces - .map((n) => n.replace(/\s+/g, " ").trim()); - // combine lines that are in parenthesis - const lines = []; - let insertSameLine = false; - cleanedLines.forEach((n) => { - if (lines.length > 0){ - if((n[0] == "(" && - lines[lines.length - 1].toLocaleLowerCase().indexOf(contants_1.CreateTable) == - -1) || - insertSameLine) { - if (lines.length > 0) { - insertSameLine = true; - lines[lines.length - 1] += ` ${n}`; - if (n[0] == ")") - insertSameLine = false; - } - } - else if(lines[lines.length - 1].match(/CONSTRAINT/gi) && - (n.match(/FOREIGN KEY/gi) && !n.match(/CONSTRAINT/gi)) - ){ - lines[lines.length - 1] += ` ${n}`; - } - // add to previous line if current has references and previous has foreign key - else if(lines[lines.length - 1].match(/FOREIGN KEY/gi) && - (n.match(/REFERENCES/gi) && !n.match(/FOREIGN KEY/gi)) - ){ - lines[lines.length - 1] += ` ${n}`; - } - else if(n.substring(0,2).toUpperCase() == "ON"){ - lines[lines.length - 1] += ` ${n}`; - } - else { - lines.push(n); - } - } - else { - lines.push(n); - } - }); - let currentTableModel = null; - //Parse SQL to objects - for (let i = 0; i < lines.length; i++) { - // rowCell = null; - const tmp = lines[i].trim(); - const propertyRow = tmp.toLowerCase().trim(); - if (propertyRow[0] == ")") { - // close table - if (currentTableModel) { - this.tableList.push(currentTableModel); - currentTableModel = null; - } - continue; - } - //Parse Table - if (propertyRow.indexOf(contants_1.CreateTable) != -1) { - //Parse row - let name = tmp - .replace(this.stringToRegex(`/${contants_1.CreateTable}/gi`), "") - .trim(); - //Parse Table Name - name = this.ParseTableName(name); - if (currentTableModel !== null) { - //Add table to the list - this.tableList.push(currentTableModel); - } - //Create Table - currentTableModel = this.CreateTable(name); - } - // Parse Properties - else if (tmp !== "(" && - currentTableModel != null && - propertyRow.indexOf(contants_1.AlterTable) == -1) { - //Parse the row - let name = tmp.substring(0, tmp.charAt(tmp.length - 1) === "," ? tmp.length - 1 : tmp.length); - //Attempt to get the Key Type - let propertyType = name.toLowerCase(); - // .substring(0, AlterTable.length).toLowerCase(); - //Add special constraints - if (this.MODE_SQLSERVER) { - if (propertyType.indexOf(contants_1.CONSTRAINT) !== -1 && - propertyType.indexOf(contants_1.Primary_Key) !== -1) { - propertyType = contants_1.CONSTRAINT_Primary_Key; - } - if (propertyType.indexOf(contants_1.CONSTRAINT) !== -1 && - propertyType.indexOf(contants_1.Foreign_Key) !== -1) { - propertyType = contants_1.CONSTRAINT_Foreign_Key; - } - } - //Verify if this is a property that doesn't have a relationship (One minute of silence for the property) - const normalProperty = !propertyType.match(/PRIMARY KEY\s?\(/gi) && - propertyType.indexOf(contants_1.Foreign_Key) == -1 && - propertyType.indexOf(contants_1.CONSTRAINT_Primary_Key) == -1 && - propertyType.indexOf(contants_1.CONSTRAINT_Foreign_Key) == -1; - const nameSkipCheck = name.toUpperCase().trim(); - //Parse properties that don't have relationships - if (normalProperty) { - if (name === "" || name === "" || name === ");") { - continue; - } - let ExtendedProperties = null; - if (this.MODE_SQLSERVER) { - if (nameSkipCheck.indexOf(" ASC") !== -1 || - nameSkipCheck.indexOf(" DESC") !== -1 || - nameSkipCheck.indexOf("EXEC ") !== -1 || - nameSkipCheck.indexOf("WITH ") !== -1 || - nameSkipCheck.indexOf(" ON") !== -1 || - nameSkipCheck.indexOf("ALTER ") !== -1 || - // comments already removed - nameSkipCheck.indexOf("/*") !== -1 || - nameSkipCheck.indexOf(" CONSTRAINT") !== -1 || - nameSkipCheck.indexOf("SET ") !== -1 || - nameSkipCheck.indexOf(" NONCLUSTERED") !== -1 || - // no spaces desired - nameSkipCheck.indexOf("GO") !== -1 || - nameSkipCheck.indexOf("REFERENCES ") !== -1) { - continue; - } - //Get delimiter of column name - const firstSpaceIndex = name[0] == "[" && name.indexOf("]" + " ") !== -1 - ? name.indexOf("]" + " ") - : name.indexOf(" "); - ExtendedProperties = name.substring(firstSpaceIndex + 1).trim(); - //Get full name - name = name.substring(0, firstSpaceIndex); - name = this.RemoveNameQuantifiers(name); - } - else { - const columnQuantifiers = this.GetColumnQuantifiers(); - //Get delimiter of column name - const firstSpaceIndex = name[0] == columnQuantifiers.Start && - name.indexOf(columnQuantifiers.End + " ") !== -1 - ? name.indexOf(columnQuantifiers.End + " ") - : name.indexOf(" "); - ExtendedProperties = name.substring(firstSpaceIndex + 1).trim(); - //Get full name - name = name.substring(0, firstSpaceIndex); - name = this.RemoveNameQuantifiers(name); - } - //Create Property - const propertyModel = this.CreateProperty(name, currentTableModel.Name, null, false, ExtendedProperties); - //Add Property to table - currentTableModel.Properties.push(propertyModel); - if (ExtendedProperties.toLocaleLowerCase().indexOf(contants_1.Primary_Key) > -1) { - //Create Primary Key - const primaryKeyModel = this.CreatePrimaryKey(name, currentTableModel.Name); - //Add Primary Key to List - this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); - } - } - else { - //Parse Primary Key - if (propertyType.indexOf(contants_1.Primary_Key) != -1 || - propertyType.indexOf(contants_1.CONSTRAINT_Primary_Key) != -1) { - if (!this.MODE_SQLSERVER) { - const primaryKey = name - .replace(/PRIMARY KEY\s?\(/gi, "") - .replace(")", ""); - //Create Primary Key - const primaryKeyModel = this.CreatePrimaryKey(primaryKey, currentTableModel.Name); - //Add Primary Key to List - this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); - } - else { - if (propertyRow.indexOf(contants_1.Primary_Key) !== -1 && - nameSkipCheck.indexOf("CLUSTERED") === -1) { - const primaryKey = name - .replace(/PRIMARY KEY\s?\(/gi, "") - .replace(")", ""); - //Create Primary Key - const primaryKeyModel = this.CreatePrimaryKey(primaryKey, currentTableModel.Name); - //Add Primary Key to List - this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); - } - else { - const startIndex = name.toLocaleLowerCase().indexOf("("); - const endIndex = name.indexOf(")") + 1; - const primaryKey = name - .substring(startIndex, endIndex) - .replace("(", "") - .replace(")", "") - .replace(/ASC/gi, "") - .trim(); - const columnQuantifiers = this.GetColumnQuantifiers(); - //Get delimiter of column name - const firstSpaceIndex = primaryKey[0] == columnQuantifiers.Start && - primaryKey.indexOf(columnQuantifiers.End + " ") !== -1 - ? primaryKey.indexOf(columnQuantifiers.End + " ") - : primaryKey.indexOf(" "); - const primaryKeyRow = firstSpaceIndex == -1 - ? primaryKey - : primaryKey.substring(firstSpaceIndex + 1).trim(); - //Create Primary Key - const primaryKeyModel = this.CreatePrimaryKey(primaryKeyRow, currentTableModel.Name); - //Add Primary Key to List - this.primaryKeyList = this.primaryKeyList.concat(primaryKeyModel); - } - } - } - } - //Parse Foreign Key - if (propertyType.indexOf(contants_1.Foreign_Key) != -1 || - propertyType.indexOf(contants_1.CONSTRAINT_Foreign_Key) != -1) { - if (!this.MODE_SQLSERVER || propertyRow.indexOf(contants_1.AlterTable) == -1) { - this.ParseMySQLForeignKey(name, currentTableModel); - } - else { - let completeRow = name; - if (nameSkipCheck.indexOf("REFERENCES") === -1) { - const referencesRow = lines[i + 1].trim(); - completeRow = - "ALTER TABLE [dbo].[" + - currentTableModel.Name + - "] WITH CHECK ADD" + - " " + - name + - " " + - referencesRow; - } - this.ParseSQLServerForeignKey(completeRow, currentTableModel); - } - } - } - else if (propertyRow.indexOf(contants_1.AlterTable) != -1) { - if (this.MODE_SQLSERVER) { - //Parse the row - const alterTableRow = tmp.substring(0, tmp.charAt(tmp.length - 1) === "," ? tmp.length - 1 : tmp.length); - const referencesRow = lines[i + 1].trim(); - const completeRow = alterTableRow + " " + referencesRow; - this.ParseSQLServerForeignKey(completeRow, currentTableModel); - } - } - } - // parse fk and primary keys - if (this.primaryKeyList.length > 0) { - this.primaryKeyList.forEach((pk) => { - // find table index - const pkTableIndex = this.tableList.findIndex((t) => t.Name.toLocaleLowerCase() == - pk.PrimaryKeyTableName.toLocaleLowerCase()); - // find property index - if (pkTableIndex > -1) { - const propertyIndex = this.tableList[pkTableIndex].Properties.findIndex((p) => p.Name.toLocaleLowerCase() == - pk.PrimaryKeyName.toLocaleLowerCase()); - if (propertyIndex > -1) { - this.tableList[pkTableIndex].Properties[propertyIndex].IsPrimaryKey = true; - } - } - }); - } - if (this.foreignKeyList.length > 0) { - this.foreignKeyList.forEach((fk) => { - // find table index - const pkTableIndex = this.tableList.findIndex((t) => t.Name.toLocaleLowerCase() == - fk.ReferencesTableName.toLocaleLowerCase()); - // find property index - if (pkTableIndex > -1) { - const propertyIndex = this.tableList[pkTableIndex].Properties.findIndex((p) => p.Name.toLocaleLowerCase() == - fk.PrimaryKeyName.toLocaleLowerCase()); - if (propertyIndex > -1) { - this.tableList[pkTableIndex].Properties[propertyIndex].ForeignKey.push(fk); - if (!fk.IsDestination) { - this.tableList[pkTableIndex].Properties[propertyIndex].IsForeignKey = true; - } - } - } - }); - } - return this; - } - stringToRegex(str) { - // Main regex - const mainResult = str.match(/\/(.+)\/.*/); - const optionsResult = str.match(/\/.+\/(.*)/); - if (mainResult && optionsResult) { - const main = mainResult[1]; - // Regex options - const options = optionsResult[1]; - // Compiled regex - return new RegExp(main, options); - } - return new RegExp("//(.+)/.*/", "//.+/(.*)/"); - } - CreatePrimaryKey(primaryKeyName, primaryKeyTableName) { - const primaryKeyNames = this.RemoveNameQuantifiers(primaryKeyName) - .split(",") - .filter((n) => n) - // remove multiple spaces - .map((n) => n.replace(/\s+/g, " ").trim()); - const primaryKeys = []; - primaryKeyNames.forEach(name => { - const primaryKey = { - PrimaryKeyTableName: primaryKeyTableName, - PrimaryKeyName: name, - }; - primaryKeys.push(primaryKey); - }); - return primaryKeys; - } - CreateProperty(name, tableName, foreignKey, isPrimaryKey, columnProps) { - const isForeignKey = foreignKey !== undefined && foreignKey !== null; - const property = { - Name: name, - ColumnProperties: columnProps, - TableName: tableName, - ForeignKey: foreignKey || [], - IsForeignKey: isForeignKey, - IsPrimaryKey: isPrimaryKey, - }; - return property; - } - ParseMySQLForeignKey(name, currentTableModel) { - const referencesIndex = name.toLowerCase().indexOf("references"); - let foreignKeySQL = name.substring(0, referencesIndex); - foreignKeySQL = foreignKeySQL.substring(foreignKeySQL.toUpperCase().indexOf("FOREIGN KEY")) - let referencesSQL = name.substring(referencesIndex, name.length); - //Remove references syntax - referencesSQL = referencesSQL.replace(/REFERENCES /gi, ""); - //Get Table and Property Index - const referencedTableIndex = referencesSQL.indexOf("("); - const referencedPropertyIndex = referencesSQL.indexOf(")"); - //Get Referenced Table - const referencedTableName = referencesSQL.substring(0, referencedTableIndex); - //Get Referenced Key - const referencedPropertyName = referencesSQL.substring(referencedTableIndex + 1, referencedPropertyIndex); - //Get ForeignKey - const foreignKey = foreignKeySQL - .replace(/FOREIGN KEY\s?\(/gi, "") - .replace(")", "") - .replace(" ", ""); - //Create ForeignKey - const foreignKeyOriginModel = this.CreateForeignKey(foreignKey, currentTableModel.Name, referencedPropertyName, referencedTableName, true); - //Add ForeignKey Origin - this.foreignKeyList.push(foreignKeyOriginModel); - //Create ForeignKey - const foreignKeyDestinationModel = this.CreateForeignKey(referencedPropertyName, referencedTableName, foreignKey, currentTableModel.Name, false); - //Add ForeignKey Destination - this.foreignKeyList.push(foreignKeyDestinationModel); - } - ParseSQLServerForeignKey(name, currentTableModel) { - const referencesIndex = name.toLowerCase().indexOf("references"); - let foreignKeySQL = ""; - if (name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`) !== -1) { - foreignKeySQL = name - .substring(name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`), referencesIndex) - .replace(/FOREIGN KEY\(/gi, "") - .replace(")", ""); - } - else { - foreignKeySQL = name - .substring(name.toLowerCase().indexOf(`${contants_1.Foreign_Key}(`), referencesIndex) - .replace(/FOREIGN KEY\s?\(/gi, "") - .replace(")", ""); - } - let referencesSQL = name.substring(referencesIndex, name.length); - const nameSkipCheck = name.toUpperCase().trim(); - let alterTableName = name - .substring(0, nameSkipCheck.indexOf("WITH")) - .replace(/ALTER TABLE /gi, ""); - if (referencesIndex !== -1 && - alterTableName !== "" && - foreignKeySQL !== "" && - referencesSQL !== "") { - //Remove references syntax - referencesSQL = referencesSQL.replace(/REFERENCES /gi, ""); - //Get Table and Property Index - const referencedTableIndex = referencesSQL.indexOf("("); - const referencedPropertyIndex = referencesSQL.indexOf(")"); - //Get Referenced Table - let referencedTableName = referencesSQL.substring(0, referencedTableIndex); - //Parse Name - referencedTableName = this.ParseSQLServerName(referencedTableName); - //Get Referenced Key - let referencedPropertyName = referencesSQL.substring(referencedTableIndex + 1, referencedPropertyIndex); - //Parse Name - referencedPropertyName = this.ParseSQLServerName(referencedPropertyName); - //Get ForeignKey - let foreignKey = foreignKeySQL - .replace(/FOREIGN KEY\s?\(/gi, "") - .replace(")", ""); - //Parse Name - foreignKey = this.ParseSQLServerName(foreignKey); - //Parse Name - alterTableName = this.ParseSQLServerName(alterTableName); - //Create ForeignKey - const foreignKeyOriginModel = this.CreateForeignKey(foreignKey, alterTableName, referencedPropertyName, referencedTableName, true); - //Add ForeignKey Origin - this.foreignKeyList.push(foreignKeyOriginModel); - //Create ForeignKey - const foreignKeyDestinationModel = this.CreateForeignKey(referencedPropertyName, referencedTableName, foreignKey, alterTableName, false); - //Add ForeignKey Destination - this.foreignKeyList.push(foreignKeyDestinationModel); - } - } - CreateForeignKey(primaryKeyName, primaryKeyTableName, referencesPropertyName, referencesTableName, isDestination) { - const foreignKey = { - PrimaryKeyTableName: this.RemoveNameQuantifiers(primaryKeyTableName), - PrimaryKeyName: this.RemoveNameQuantifiers(primaryKeyName), - ReferencesPropertyName: this.RemoveNameQuantifiers(referencesPropertyName), - ReferencesTableName: this.RemoveNameQuantifiers(referencesTableName), - IsDestination: isDestination !== undefined && isDestination !== null - ? isDestination - : false, - }; - return foreignKey; - } - RemoveNameQuantifiers(name) { - return name.replace(/\[|\]|\(|\"|\'|\`/g, "").trim(); - } - ParseTableName(name) { - if (name.charAt(name.length - 1) === "(") { - name = this.RemoveNameQuantifiers(name); - } - return name; - } - ParseSQLServerName(name, property) { - name = name.replace("[dbo].[", ""); - name = name.replace("](", ""); - name = name.replace("].[", "."); - name = name.replace("[", ""); - if (property == undefined || property == null) { - name = name.replace(" [", ""); - name = name.replace("] ", ""); - } - else { - if (name.indexOf("]") !== -1) { - name = name.substring(0, name.indexOf("]")); - } - } - if (name.lastIndexOf("]") === name.length - 1) { - name = name.substring(0, name.length - 1); - } - if (name.lastIndexOf(")") === name.length - 1) { - name = name.substring(0, name.length - 1); - } - if (name.lastIndexOf("(") === name.length - 1) { - name = name.substring(0, name.length - 1); - } - name = name.replace(" ", ""); - return name; - } - CreateTable(name) { - const table = { - Name: name, - Properties: [], - }; - //Count exported tables - this.exportedTables++; - return table; - } - /** - * Checks whether character is a quotation character. - * - * @param char Character to be evaluated. - */ - static isQuoteChar(char) { - return char === "\"" || char === "'" || char === "`"; - } - /** - * convert labels with start and end strings per database type - * @param label - * @returns - */ - dbTypeEnds(label) { - let char1 = "\""; - let char2 = "\""; - if (this.dialect == "mysql") { - char1 = "`"; - char2 = "`"; - } - else if (this.dialect == "sqlserver") { - char1 = "["; - char2 = "]"; - } - return `${char1}${label}${char2}`; - } - WithEnds() { - this.tableList = this.tableList.map((table) => { - table.Name = this.dbTypeEnds(table.Name); - table.Properties = table.Properties.map((property) => { - property.Name = this.dbTypeEnds(property.Name); - property.TableName = this.dbTypeEnds(property.TableName); - return property; - }); - return table; - }); - this.primaryKeyList = this.primaryKeyList.map((primaryKey) => { - primaryKey.PrimaryKeyName = this.dbTypeEnds(primaryKey.PrimaryKeyName); - primaryKey.PrimaryKeyTableName = this.dbTypeEnds(primaryKey.PrimaryKeyTableName); - return primaryKey; - }); - this.foreignKeyList = this.foreignKeyList.map((foreignKey) => { - foreignKey.PrimaryKeyName = this.dbTypeEnds(foreignKey.PrimaryKeyName); - foreignKey.ReferencesPropertyName = this.dbTypeEnds(foreignKey.ReferencesPropertyName); - foreignKey.PrimaryKeyTableName = this.dbTypeEnds(foreignKey.PrimaryKeyTableName); - foreignKey.ReferencesTableName = this.dbTypeEnds(foreignKey.ReferencesTableName); - return foreignKey; - }); - return this; - } - WithoutEnds() { - this.tableList.map((table) => { - table.Name = this.RemoveNameQuantifiers(table.Name); - table.Properties = table.Properties.map((property) => { - property.Name = this.RemoveNameQuantifiers(property.Name); - property.TableName = this.RemoveNameQuantifiers(property.TableName); - return property; - }); - return table; - }); - this.primaryKeyList = this.primaryKeyList.map((primaryKey) => { - primaryKey.PrimaryKeyName = this.RemoveNameQuantifiers(primaryKey.PrimaryKeyName); - primaryKey.PrimaryKeyTableName = this.RemoveNameQuantifiers(primaryKey.PrimaryKeyTableName); - return primaryKey; - }); - this.foreignKeyList = this.foreignKeyList.map((foreignKey) => { - foreignKey.PrimaryKeyName = this.RemoveNameQuantifiers(foreignKey.PrimaryKeyName); - foreignKey.ReferencesPropertyName = this.RemoveNameQuantifiers(foreignKey.ReferencesPropertyName); - foreignKey.PrimaryKeyTableName = this.RemoveNameQuantifiers(foreignKey.PrimaryKeyTableName); - foreignKey.ReferencesTableName = this.RemoveNameQuantifiers(foreignKey.ReferencesTableName); - return foreignKey; - }); - return this; - } - /** - * return text quantifiers for dialect - * @returns json - */ - GetColumnQuantifiers() { - const chars = { - Start: "\"", - End: "\"", - }; - if (this.dialect == "mysql") { - chars.Start = "`"; - chars.End = "`"; - } - else if (this.dialect == "sqlserver") { - chars.Start = "["; - chars.End = "]"; - } - return chars; - } - ToTableList() { - return this.tableList; - } - ToPrimaryKeyList() { - return this.primaryKeyList; - } - ToForeignKeyList() { - return this.foreignKeyList; - } - ResetModel() { - this.tableList = []; - this.primaryKeyList = []; - this.foreignKeyList = []; - return this; - } - /** - * return full sql model - * @returns - */ - ToModel() { - return { - TableList: this.tableList, - Dialect: this.dialect, - ForeignKeyList: this.foreignKeyList, - PrimaryKeyList: this.primaryKeyList, - }; - } - } - //Table Info - var foreignKeyList = []; - var primaryKeyList = []; - var tableList = []; - var cells = []; - var tableCell = null; - var rowCell = null; - var dx = 0; - var exportedTables = 0; - - + let foreignKeyList = []; + let primaryKeyList = []; + let tableList = []; + let cells = []; + let tableCell = null; + let rowCell = null; + let dx = 0; + let exportedTables = 0; //Create Base div - const divFromSQL = document.createElement('div'); - divFromSQL.style.userSelect = 'none'; - divFromSQL.style.overflow = 'hidden'; - divFromSQL.style.padding = '10px'; - divFromSQL.style.height = '100%'; - + const divFromSQL = document.createElement("div"); + divFromSQL.style.userSelect = "none"; + divFromSQL.style.overflow = "hidden"; + divFromSQL.style.padding = "10px"; + divFromSQL.style.height = "100%"; var graph = ui.editor.graph; - - const sqlInputFromSQL = document.createElement('textarea'); - sqlInputFromSQL.style.height = '200px'; - sqlInputFromSQL.style.width = '100%'; - var defaultReset = '/*\n\tDraw io default value\n*/\n\nCREATE TABLE Persons\n(\n PersonID int NOT NULL,\n LastName varchar(255),\n ' + - 'FirstName varchar(255),\n Address varchar(255),\n City varchar(255),\n Primary Key(PersonID)\n);\n\n' + - 'CREATE TABLE Orders\n(\n OrderID int NOT NULL PRIMARY KEY,\n PersonID int NOT NULL,\n FOREIGN KEY ([PersonID]) REFERENCES [Persons]([PersonID])' + - '\n);' - sqlInputFromSQL.value = defaultReset + const sqlInputFromSQL = document.createElement("textarea"); + sqlInputFromSQL.style.height = "200px"; + sqlInputFromSQL.style.width = "100%"; + const defaultReset = "/*\n\tDrawio default value\n\tPlugin: sql\n\tVersion: ${pluginVersion}\n*/\n\nCREATE TABLE Persons\n(\n PersonID int NOT NULL,\n LastName varchar(255),\n " + + "FirstName varchar(255),\n Address varchar(255),\n City varchar(255),\n Primary Key(PersonID)\n);\n\n" + + "CREATE TABLE Orders\n(\n OrderID int NOT NULL PRIMARY KEY,\n PersonID int NOT NULL,\n FOREIGN KEY ([PersonID]) REFERENCES [Persons]([PersonID])" + + "\n);"; + sqlInputFromSQL.value = defaultReset; mxUtils.br(divFromSQL); divFromSQL.appendChild(sqlInputFromSQL); - var graph = ui.editor.graph; - // Extends Extras menu - mxResources.parse('fromSql=From SQL'); - - const wndFromSQL = new mxWindow(mxResources.get('fromSql'), divFromSQL, document.body.offsetWidth - 480, 140, - 320, 320, true, true); + mxResources.parse("fromSql=From SQL"); + const wndFromSQL = new mxWindow(mxResources.get("fromSql"), divFromSQL, document.body.offsetWidth - 480, 140, 320, 320, true, true); wndFromSQL.destroyOnClose = false; wndFromSQL.setMaximizable(false); wndFromSQL.setResizable(false); wndFromSQL.setClosable(true); - function AddRow(propertyModel, tableName) { - - var cellName = propertyModel.Name + (propertyModel.ColumnProperties ? " " + propertyModel.ColumnProperties: ""); - - rowCell = new mxCell(cellName, new mxGeometry(0, 0, 90, 26), - 'shape=partialRectangle;top=0;left=0;right=0;bottom=0;align=left;verticalAlign=top;spacingTop=-2;fillColor=none;spacingLeft=64;spacingRight=4;overflow=hidden;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;dropTarget=0;'); + const cellName = propertyModel.Name + (propertyModel.ColumnProperties ? " " + propertyModel.ColumnProperties : ""); + rowCell = new mxCell(cellName, new mxGeometry(0, 0, 90, 26), "shape=partialRectangle;top=0;left=0;right=0;bottom=0;align=left;verticalAlign=top;spacingTop=-2;fillColor=none;spacingLeft=64;spacingRight=4;overflow=hidden;rotatable=0;points=[[0,0.5],[1,0.5]];portConstraint=eastwest;dropTarget=0;"); rowCell.vertex = true; - - var columnType = propertyModel.IsPrimaryKey && propertyModel.IsForeignKey ? 'PK | FK' : propertyModel.IsPrimaryKey ? 'PK' : propertyModel.IsForeignKey ? 'FK' : ''; - - var left = sb.cloneCell(rowCell, columnType); + const columnType = propertyModel.IsPrimaryKey && propertyModel.IsForeignKey ? "PK | FK" : propertyModel.IsPrimaryKey ? "PK" : propertyModel.IsForeignKey ? "FK" : ""; + const left = sb.cloneCell(rowCell, columnType); left.connectable = false; - left.style = 'shape=partialRectangle;top=0;left=0;bottom=0;fillColor=none;align=left;verticalAlign=middle;spacingLeft=4;spacingRight=4;overflow=hidden;rotatable=180;points=[];portConstraint=eastwest;part=1;' + left.style = "shape=partialRectangle;top=0;left=0;bottom=0;fillColor=none;align=left;verticalAlign=middle;spacingLeft=4;spacingRight=4;overflow=hidden;rotatable=180;points=[];portConstraint=eastwest;part=1;"; left.geometry.width = 54; left.geometry.height = 26; rowCell.insert(left); - - var size = ui.editor.graph.getPreferredSizeForCell(rowCell); - - if (size !== null && tableCell.geometry.width < size.width + 10) { - tableCell.geometry.width = size.width + 10; + const size = ui.editor.graph.getPreferredSizeForCell(rowCell); + if (tableCell) { + if (size !== null && tableCell.geometry.width < size.width + 10) { + tableCell.geometry.width = size.width + 10; + } + tableCell.insert(rowCell); + tableCell.geometry.height += 26; } - - tableCell.insert(rowCell); - tableCell.geometry.height += 26; - rowCell = rowCell; - - }; - + } + ; function parseSql(text, type) { // reset values - cells = [] + cells = []; tableCell = null; rowCell = null; // load parser - const parser = new SqlSimpleParser(type); - - + const parser = new sqlsimpleparser_1.SqlSimpleParser(type); const models = parser .feed(text) .WithoutEnds() .WithEnds() .ToModel(); - - - foreignKeyList = models.ForeignKeyList; primaryKeyList = models.PrimaryKeyList; tableList = models.TableList; exportedTables = tableList.length; - //Create Table in UI CreateTableUI(type); - }; - /** - * return text quantifiers for dialect - * @returns json - */ - function GetColumnQuantifiers(type) { - let chars = { - Start: '"', - End: '"', - }; - if (type == "mysql") { - chars.Start = "`"; - chars.End = "`"; - } - else if (type == "sqlserver") { - chars.Start = "["; - chars.End = "]"; - } - return chars; - } - - /** - * extract row column attributes - * @param {*} label - * @param {*} columnQuantifiers - * @returns - */ - function getDbLabel(label, columnQuantifiers){ - // fix duplicate spaces and different space chars - label = label - .replace(/\s+/g, " ") - let firstSpaceIndex = label[0] == columnQuantifiers.Start && - label.indexOf(columnQuantifiers.End + " ") !== -1 - ? label.indexOf(columnQuantifiers.End + " ") - : label.indexOf(" "); - let attributeType = label.substring(firstSpaceIndex + 1).trim(); - let attributeName = label.substring(0, firstSpaceIndex + 1); - let attribute = { - attributeName, - attributeType - } - return attribute } - + ; function CreateTableUI(type) { - tableList.forEach(function(tableModel) { + tableList.forEach(function (tableModel) { //Define table size width - var maxNameLenght = 100 + tableModel.Name.length; - + const maxNameLenght = 100 + tableModel.Name.length; //Create Table - tableCell = new mxCell(tableModel.Name, new mxGeometry(dx, 0, maxNameLenght, 26), - 'swimlane;fontStyle=0;childLayout=stackLayout;horizontal=1;startSize=26;fillColor=default;horizontalStack=0;resizeParent=1;resizeLast=0;collapsible=1;marginBottom=0;swimlaneFillColor=default;align=center;'); + tableCell = new mxCell(tableModel.Name, new mxGeometry(dx, 0, maxNameLenght, 26), "swimlane;fontStyle=0;childLayout=stackLayout;horizontal=1;startSize=26;fillColor=default;horizontalStack=0;resizeParent=1;resizeLast=0;collapsible=1;marginBottom=0;swimlaneFillColor=default;align=center;"); tableCell.vertex = true; - //Resize row - var size = ui.editor.graph.getPreferredSizeForCell(rowCell); - if (size !== null) { - tableCell.geometry.width = size.width + maxNameLenght; + if (rowCell) { + const size = ui.editor.graph.getPreferredSizeForCell(rowCell); + if (size !== null) { + tableCell.geometry.width = size.width + maxNameLenght; + } } - //Add Table to cells cells.push(tableCell); - //Add properties - tableModel.Properties.forEach(function(propertyModel) { - + tableModel.Properties.forEach(function (propertyModel) { //Add row AddRow(propertyModel, tableModel.Name); }); - //Close table dx += tableCell.geometry.width + 40; tableCell = null; }); - if (cells.length > 0) { - var graph = ui.editor.graph; - var view = graph.view; - var bds = graph.getGraphBounds(); - + const graph = ui.editor.graph; + const view = graph.view; + const bds = graph.getGraphBounds(); // Computes unscaled, untranslated graph bounds - var x = Math.ceil(Math.max(0, bds.x / view.scale - view.translate.x) + 4 * graph.gridSize); - var y = Math.ceil(Math.max(0, (bds.y + bds.height) / view.scale - view.translate.y) + 4 * graph.gridSize); - + const x = Math.ceil(Math.max(0, bds.x / view.scale - view.translate.x) + 4 * graph.gridSize); + const y = Math.ceil(Math.max(0, (bds.y + bds.height) / view.scale - view.translate.y) + 4 * graph.gridSize); graph.setSelectionCells(graph.importCells(cells, x, y)); // add foreign key edges - var model = graph.getModel(); + const model = graph.getModel(); const columnQuantifiers = GetColumnQuantifiers(type); - var pt = graph.getFreeInsertPoint(); - foreignKeyList.forEach(function(fk){ - if(fk.IsDestination && fk.PrimaryKeyName && fk.ReferencesPropertyName && + const pt = graph.getFreeInsertPoint(); + foreignKeyList.forEach(function (fk) { + if (fk.IsDestination && fk.PrimaryKeyName && fk.ReferencesPropertyName && fk.PrimaryKeyTableName && fk.ReferencesTableName) { - var insertEdge = mxUtils.bind(this, function(targetCell, sourceCell, edge){ - var label = "" - var edgeStyle = "edgeStyle=entityRelationEdgeStyle;html=1;endArrow=ERzeroToMany;startArrow=ERzeroToOne;labelBackgroundColor=none;fontFamily=Verdana;fontSize=14;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=-0.018;entryY=0.608;entryDx=0;entryDy=0;entryPerimeter=0;" - var edgeCell = graph.insertEdge(null, null, label || '', (edge.invert) ? - sourceCell : targetCell, (edge.invert) ? targetCell : sourceCell, edgeStyle); + const insertEdge = mxUtils.bind(this, function (targetCell, sourceCell, edge) { + const label = ""; + const edgeStyle = "edgeStyle=entityRelationEdgeStyle;html=1;endArrow=ERzeroToMany;startArrow=ERzeroToOne;labelBackgroundColor=none;fontFamily=Verdana;fontSize=14;exitX=1;exitY=0.5;exitDx=0;exitDy=0;entryX=-0.018;entryY=0.608;entryDx=0;entryDy=0;entryPerimeter=0;"; + const edgeCell = graph.insertEdge(null, null, label || "", (edge.invert) ? + sourceCell : targetCell, (edge.invert) ? targetCell : sourceCell, edgeStyle); }); - let edge = { + const edge = { invert: true }; - var targetCell = null; - var sourceCell = null; + let targetCell = null; + let sourceCell = null; // locate edge source and target cells for (const key in model.cells) { - if(targetCell && sourceCell) + if (targetCell && sourceCell) break; if (Object.hasOwnProperty.call(model.cells, key)) { const mxcell = model.cells[key]; - if(mxcell.style && mxcell.style.trim().startsWith("swimlane;")){ - let entity = { + if (mxcell.style && mxcell.style.trim().startsWith("swimlane;")) { + const entity = { name: mxcell.value, attributes: [] - } - var isPrimaryTable = entity.name == fk.PrimaryKeyTableName; - var isForeignTable = entity.name == fk.ReferencesTableName; - if(isPrimaryTable || isForeignTable){ + }; + const isPrimaryTable = entity.name == fk.PrimaryKeyTableName; + const isForeignTable = entity.name == fk.ReferencesTableName; + if (isPrimaryTable || isForeignTable) { for (let c = 0; c < mxcell.children.length; c++) { - if(targetCell && sourceCell) + if (targetCell && sourceCell) break; const col = mxcell.children[c]; - if(col.mxObjectId.indexOf("mxCell") !== -1) { - if(col.style && col.style.trim().startsWith("shape=partialRectangle")){ - let attribute = getDbLabel(col.value, columnQuantifiers) - if(isPrimaryTable && attribute.attributeName == fk.PrimaryKeyName){ + if (col.mxObjectId.indexOf("mxCell") !== -1) { + if (col.style && col.style.trim().startsWith("shape=partialRectangle")) { + const attribute = getDbLabel(col.value, columnQuantifiers); + if (isPrimaryTable && attribute.attributeName == fk.PrimaryKeyName) { targetCell = col; break; - } else if(isForeignTable && attribute.attributeName == fk.ReferencesPropertyName){ + } + else if (isForeignTable && attribute.attributeName == fk.ReferencesPropertyName) { sourceCell = col; break; } @@ -1364,96 +1384,88 @@ Draw.loadPlugin(function(ui) { } } } - } } } - if(targetCell && sourceCell) + if (targetCell && sourceCell) insertEdge(targetCell, sourceCell, edge); } - }) + }); graph.scrollCellToVisible(graph.getSelectionCell()); } - wndFromSQL.setVisible(false); - }; - + } + ; mxUtils.br(divFromSQL); - - const resetBtnFromSQL = mxUtils.button(mxResources.get('reset'), function() { + const resetBtnFromSQL = mxUtils.button(mxResources.get("reset"), function () { sqlInputFromSQL.value = defaultReset; }); - - resetBtnFromSQL.style.marginTop = '8px'; - resetBtnFromSQL.style.marginRight = '4px'; - resetBtnFromSQL.style.padding = '4px'; + resetBtnFromSQL.style.marginTop = "8px"; + resetBtnFromSQL.style.marginRight = "4px"; + resetBtnFromSQL.style.padding = "4px"; divFromSQL.appendChild(resetBtnFromSQL); - - const btnFromSQL_mysql = mxUtils.button('Insert MySQL', function() { - parseSql(sqlInputFromSQL.value, 'mysql'); + const btnFromSQL_mysql = mxUtils.button("Insert MySQL", function () { + parseSql(sqlInputFromSQL.value, "mysql"); }); - - btnFromSQL_mysql.style.marginTop = '8px'; - btnFromSQL_mysql.style.padding = '4px'; + btnFromSQL_mysql.style.marginTop = "8px"; + btnFromSQL_mysql.style.padding = "4px"; divFromSQL.appendChild(btnFromSQL_mysql); - - const btnFromSQL_sqlserver = mxUtils.button('Insert SQL Server', function() { - parseSql(sqlInputFromSQL.value, 'sqlserver'); + const btnFromSQL_sqlserver = mxUtils.button("Insert SQL Server", function () { + parseSql(sqlInputFromSQL.value, "sqlserver"); }); - - btnFromSQL_sqlserver.style.marginTop = '8px'; - btnFromSQL_sqlserver.style.padding = '4px'; + btnFromSQL_sqlserver.style.marginTop = "8px"; + btnFromSQL_sqlserver.style.padding = "4px"; divFromSQL.appendChild(btnFromSQL_sqlserver); - - const btnFromSQL_postgres = mxUtils.button('Insert PostgreSQL', function() { - parseSql(sqlInputFromSQL.value, 'postgres'); + const btnFromSQL_postgres = mxUtils.button("Insert PostgreSQL", function () { + parseSql(sqlInputFromSQL.value, "postgres"); }); - - btnFromSQL_postgres.style.marginTop = '8px'; - btnFromSQL_postgres.style.padding = '4px'; + btnFromSQL_postgres.style.marginTop = "8px"; + btnFromSQL_postgres.style.padding = "4px"; divFromSQL.appendChild(btnFromSQL_postgres); - - const btnFromSQL_sqlite = mxUtils.button('Insert Sqlite', function() { - parseSql(sqlInputFromSQL.value, 'sqlite'); + const btnFromSQL_sqlite = mxUtils.button("Insert Sqlite", function () { + parseSql(sqlInputFromSQL.value, "sqlite"); }); - - btnFromSQL_sqlite.style.marginTop = '8px'; - btnFromSQL_sqlite.style.padding = '4px'; + btnFromSQL_sqlite.style.marginTop = "8px"; + btnFromSQL_sqlite.style.padding = "4px"; divFromSQL.appendChild(btnFromSQL_sqlite); - // Adds action - ui.actions.addAction('fromSql', function() { + ui.actions.addAction("fromSql", function () { wndFromSQL.setVisible(!wndFromSQL.isVisible()); - if (wndFromSQL.isVisible()) { sqlInputFromSQL.focus(); } }); // end import diagrams from sql text methods - // finalize menu buttons - var theMenu = ui.menus.get('insert'); - var oldMenu = theMenu.funct; - - theMenu.funct = function(menu, parent) { - oldMenu.apply(this, arguments); - ui.menus.addMenuItems(menu, ['fromSql'], parent); - }; - if(theMenuExportAs && theMenuExportAs.enabled) { + const theMenu = ui.menus.get("insert"); + if (theMenu) { + const oldMenu = theMenu.funct; + theMenu.funct = function (...args) { + const [menu, parent] = args; + oldMenu.apply(this, args); + ui.menus.addMenuItems(menu, ["fromSql"], parent); + }; + } + if (theMenuExportAs && theMenuExportAs.enabled) { var oldMenuExportAs = theMenuExportAs.funct; - - theMenuExportAs.funct = function(menu, parent) { - oldMenuExportAs.apply(this, arguments); - ui.menus.addMenuItems(menu, ['tosql'], parent); + theMenuExportAs.funct = function (...args) { + const [menu, parent] = args; + oldMenuExportAs.apply(this, args); + ui.menus.addMenuItems(menu, ["tosql"], parent); }; - } else { + } + else { // vscode file export sql menu - var menu = ui.menus.get('file'); - var oldMenuExportAs = menu.funct; - menu.funct = function(menu, parent) { - oldMenuExportAs.apply(this, arguments); - debugger - ui.menus.addMenuItems(menu, ['tosql'], parent); - }; + const menu = ui.menus.get("file"); + if (menu && menu.enabled) { + var oldMenuExportAs = menu.funct; + menu.funct = function (...args) { + const [menu, parent] = args; + oldMenuExportAs.apply(this, args); + ui.menus.addMenuItems(menu, ["tosql"], parent); + }; + } } -}); \ No newline at end of file +}); + +},{"@funktechno/little-mermaid-2-the-sql/lib/src/generate-sql-ddl":1,"@funktechno/sqlsimpleparser":3}]},{},[4]); diff --git a/src/sql.ts b/src/sql.ts index 088ec85..581944c 100644 --- a/src/sql.ts +++ b/src/sql.ts @@ -11,12 +11,7 @@ import { ForeignKeyModel, PrimaryKeyModel, PropertyModel, TableModel } from "@fu Draw.loadPlugin(function(ui) { // export sql methods - - /** - * Mermaid Models TO SQL parser - * src https://github.com/Software-Developers-IRL/Little-Mermaid-2-The-SQL/blob/main/src/generate-sql-ddl.ts - */ - // DbParser + const pluginVersion = ""; //Create Base div const divGenSQL = document.createElement("div"); @@ -28,7 +23,8 @@ Draw.loadPlugin(function(ui) { const sqlInputGenSQL = document.createElement("textarea"); sqlInputGenSQL.style.height = "200px"; sqlInputGenSQL.style.width = "100%"; - sqlInputGenSQL.value = "-- click a database type button"; + const sqlExportDefault = "-- click a database type button"; + sqlInputGenSQL.value = sqlExportDefault; mxUtils.br(divGenSQL); divGenSQL.appendChild(sqlInputGenSQL); const theMenuExportAs = ui.menus.get("exportAs"); @@ -72,9 +68,10 @@ Draw.loadPlugin(function(ui) { * @returns */ function removeHtml(label: string){ - const div = document.createElement("div"); - divGenSQL.innerHTML = label; - const text = div.textContent || div.innerText || ""; + const tempDiv = document.createElement("div"); + tempDiv.innerHTML = label; + const text = tempDiv.textContent || tempDiv.innerText || ""; + tempDiv.remove(); return text; } /** @@ -93,7 +90,7 @@ Draw.loadPlugin(function(ui) { ? label.indexOf(columnQuantifiers.End + " ") : label.indexOf(" "); const attributeType = label.substring(firstSpaceIndex + 1).trim(); - const attributeName = RemoveNameQuantifiers(label.substring(0, firstSpaceIndex)); + const attributeName = RemoveNameQuantifiers(label.substring(0, firstSpaceIndex + 1)); const attribute = { attributeName, attributeType @@ -308,7 +305,7 @@ Draw.loadPlugin(function(ui) { const parser = new DbParser(type as string, db); // generate sql let sql = parser.getSQLDataDefinition(); - sql = `/*\n\tGenerated in drawio\n\tDatabase: ${type}\n*/\n\n` + sql; + sql = `/*\n\tGenerated in drawio\n\tDatabase: ${type}\n\tPlugin: sql\n\tVersion: ${pluginVersion}\n*/\n\n` + sql; sql = sql.trim(); // update sql value in text area sqlInputGenSQL.value = sql; @@ -319,7 +316,7 @@ Draw.loadPlugin(function(ui) { mxUtils.br(divGenSQL); const resetBtnGenSQL = mxUtils.button(mxResources.get("reset"), function() { - sqlInputGenSQL.value = ""; + sqlInputGenSQL.value = sqlExportDefault; }); resetBtnGenSQL.style.marginTop = "8px"; @@ -395,10 +392,11 @@ Draw.loadPlugin(function(ui) { const sqlInputFromSQL = document.createElement("textarea"); sqlInputFromSQL.style.height = "200px"; sqlInputFromSQL.style.width = "100%"; - const defaultReset = "/*\n\tDraw io default value\n*/\n\nCREATE TABLE Persons\n(\n PersonID int NOT NULL,\n LastName varchar(255),\n " + + const defaultReset = "/*\n\tDrawio default value\n\tPlugin: sql\n\tVersion: ${pluginVersion}\n*/\n\nCREATE TABLE Persons\n(\n PersonID int NOT NULL,\n LastName varchar(255),\n " + "FirstName varchar(255),\n Address varchar(255),\n City varchar(255),\n Primary Key(PersonID)\n);\n\n" + "CREATE TABLE Orders\n(\n OrderID int NOT NULL PRIMARY KEY,\n PersonID int NOT NULL,\n FOREIGN KEY ([PersonID]) REFERENCES [Persons]([PersonID])" + "\n);"; + sqlInputFromSQL.value = defaultReset; mxUtils.br(divFromSQL); divFromSQL.appendChild(sqlInputFromSQL);