diff --git a/.vscode/settings.json b/.vscode/settings.json deleted file mode 100644 index d1fb129..0000000 --- a/.vscode/settings.json +++ /dev/null @@ -1,12 +0,0 @@ -{ - "fiveServer.ignore": [ - "**/node_modules/**", - ".vscode/**", - ".git/**", - "**/*.scss", - "**/*.sass", - "**/*.css", - "**/*.ts", - "**/*.js", - ] -} \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index 0fbdb78..c09791e 100644 --- a/package-lock.json +++ b/package-lock.json @@ -27,7 +27,9 @@ "didyoumean": "^1.2.2", "dlv": "^1.1.3", "eastasianwidth": "^0.2.0", + "ejs": "^3.1.10", "emoji-regex": "^9.2.2", + "express": "^4.21.2", "fast-glob": "^3.3.2", "fastq": "^1.18.0", "fill-range": "^7.1.1", @@ -96,6 +98,7 @@ "which": "^2.0.2", "wrap-ansi": "^8.1.0", "wrap-ansi-cjs": "^7.0.0", + "ws": "^8.18.0", "yaml": "^2.6.1" }, "devDependencies": { @@ -215,6 +218,18 @@ "node": ">=14" } }, + "node_modules/accepts": { + "version": "1.3.8", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", + "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==", + "dependencies": { + "mime-types": "~2.1.34", + "negotiator": "0.6.3" + }, + "engines": { + "node": ">= 0.6" + } + }, "node_modules/ansi-regex": { "version": "6.1.0", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.1.0.tgz", @@ -259,6 +274,16 @@ "resolved": "https://registry.npmjs.org/arg/-/arg-5.0.2.tgz", "integrity": "sha512-PYjyFOLKQ9y57JvQ6QLo8dAgNqswh8M1RMJYdQduT6xbWSgK36P/Z/v+p888pM69jMMfS8Xd8F6I1kQ/I9HUGg==" }, + "node_modules/array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==" + }, + "node_modules/async": { + "version": "3.2.6", + "resolved": "https://registry.npmjs.org/async/-/async-3.2.6.tgz", + "integrity": "sha512-htCUDlxyyCLMgaM3xXg0C0LW2xqfuQ6p05pCEIsXuyQ+a1koYKTuBMzRNwmybfLgvJDMd0r1LTn4+E0Ti6C2AA==" + }, "node_modules/balanced-match": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", @@ -275,6 +300,29 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/body-parser": { + "version": "1.20.3", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.20.3.tgz", + "integrity": "sha512-7rAxByjUMqQ3/bHJy7D6OGXvx/MMc4IqBn/X0fcM1QUcAItpZrBEYhWGem+tzXH90c+G01ypMcYJBO9Y30203g==", + "dependencies": { + "bytes": "3.1.2", + "content-type": "~1.0.5", + "debug": "2.6.9", + "depd": "2.0.0", + "destroy": "1.2.0", + "http-errors": "2.0.0", + "iconv-lite": "0.4.24", + "on-finished": "2.4.1", + "qs": "6.13.0", + "raw-body": "2.5.2", + "type-is": "~1.6.18", + "unpipe": "1.0.0" + }, + "engines": { + "node": ">= 0.8", + "npm": "1.2.8000 || >= 1.4.16" + } + }, "node_modules/brace-expansion": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", @@ -294,6 +342,41 @@ "node": ">=8" } }, + "node_modules/bytes": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", + "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/call-bind-apply-helpers": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.1.tgz", + "integrity": "sha512-BhYE+WDaywFg2TBWYNXAE+8B1ATnThNBqXHP5nQu0jWJdVvY2hvkpyB3qOmtmDePiS5/BDQ8wASEWGMWRG148g==", + "dependencies": { + "es-errors": "^1.3.0", + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/call-bound": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.3.tgz", + "integrity": "sha512-YTd+6wGlNlPxSuri7Y6X8tY2dmm12UMH66RpKMhiX6rsk5wXXnYgbUcOt8kiS31/AjfoTOvCsE+w8nZQLQnzHA==", + "dependencies": { + "call-bind-apply-helpers": "^1.0.1", + "get-intrinsic": "^1.2.6" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/camelcase-css": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/camelcase-css/-/camelcase-css-2.0.1.tgz", @@ -302,6 +385,35 @@ "node": ">= 6" } }, + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "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/chalk/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, "node_modules/chokidar": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-4.0.3.tgz", @@ -354,6 +466,43 @@ "node": ">= 6" } }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==" + }, + "node_modules/content-disposition": { + "version": "0.5.4", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.4.tgz", + "integrity": "sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==", + "dependencies": { + "safe-buffer": "5.2.1" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/content-type": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz", + "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie": { + "version": "0.7.1", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.1.tgz", + "integrity": "sha512-6DnInpx7SJ2AK3+CTUE/ZM0vWTUboZCegxhC2xiIydHR9jNuTAASBrfEpHhiGOZw/nX51bHt6YQl8jsGo4y/0w==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie-signature": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", + "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==" + }, "node_modules/cross-spawn": { "version": "7.0.6", "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", @@ -378,6 +527,31 @@ "node": ">=4" } }, + "node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/depd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", + "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/destroy": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.2.0.tgz", + "integrity": "sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==", + "engines": { + "node": ">= 0.8", + "npm": "1.2.8000 || >= 1.4.16" + } + }, "node_modules/didyoumean": { "version": "1.2.2", "resolved": "https://registry.npmjs.org/didyoumean/-/didyoumean-1.2.2.tgz", @@ -388,16 +562,141 @@ "resolved": "https://registry.npmjs.org/dlv/-/dlv-1.1.3.tgz", "integrity": "sha512-+HlytyjlPKnIG8XuRG8WvmBP8xs8P71y+SKKS6ZXWoEgLuePxtDoUEiH7WkdePWrQ5JBpE6aoVqfZfJUQkjXwA==" }, + "node_modules/dunder-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz", + "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==", + "dependencies": { + "call-bind-apply-helpers": "^1.0.1", + "es-errors": "^1.3.0", + "gopd": "^1.2.0" + }, + "engines": { + "node": ">= 0.4" + } + }, "node_modules/eastasianwidth": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==" }, + "node_modules/ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==" + }, + "node_modules/ejs": { + "version": "3.1.10", + "resolved": "https://registry.npmjs.org/ejs/-/ejs-3.1.10.tgz", + "integrity": "sha512-UeJmFfOrAQS8OJWPZ4qtgHyWExa088/MtK5UEyoJGFH67cDEXkZSviOiKRCZ4Xij0zxI3JECgYs3oKx+AizQBA==", + "dependencies": { + "jake": "^10.8.5" + }, + "bin": { + "ejs": "bin/cli.js" + }, + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/emoji-regex": { "version": "9.2.2", "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==" }, + "node_modules/encodeurl": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-2.0.0.tgz", + "integrity": "sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/es-define-property": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz", + "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==", + "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/es-object-atoms": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz", + "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==", + "dependencies": { + "es-errors": "^1.3.0" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==" + }, + "node_modules/etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/express": { + "version": "4.21.2", + "resolved": "https://registry.npmjs.org/express/-/express-4.21.2.tgz", + "integrity": "sha512-28HqgMZAmih1Czt9ny7qr6ek2qddF4FclbMzwhCREB6OFfH+rXAnuNCwo1/wFvrtbgsQDb4kSbX9de9lFbrXnA==", + "dependencies": { + "accepts": "~1.3.8", + "array-flatten": "1.1.1", + "body-parser": "1.20.3", + "content-disposition": "0.5.4", + "content-type": "~1.0.4", + "cookie": "0.7.1", + "cookie-signature": "1.0.6", + "debug": "2.6.9", + "depd": "2.0.0", + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "finalhandler": "1.3.1", + "fresh": "0.5.2", + "http-errors": "2.0.0", + "merge-descriptors": "1.0.3", + "methods": "~1.1.2", + "on-finished": "2.4.1", + "parseurl": "~1.3.3", + "path-to-regexp": "0.1.12", + "proxy-addr": "~2.0.7", + "qs": "6.13.0", + "range-parser": "~1.2.1", + "safe-buffer": "5.2.1", + "send": "0.19.0", + "serve-static": "1.16.2", + "setprototypeof": "1.2.0", + "statuses": "2.0.1", + "type-is": "~1.6.18", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + }, + "engines": { + "node": ">= 0.10.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/express" + } + }, "node_modules/fast-glob": { "version": "3.3.2", "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.2.tgz", @@ -432,6 +731,25 @@ "reusify": "^1.0.4" } }, + "node_modules/filelist": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/filelist/-/filelist-1.0.4.tgz", + "integrity": "sha512-w1cEuf3S+DrLCQL7ET6kz+gmlJdbq9J7yXCSjK/OZCPA+qEN1WyF4ZAf0YYJa4/shHJra2t/d/r8SV4Ji+x+8Q==", + "dependencies": { + "minimatch": "^5.0.1" + } + }, + "node_modules/filelist/node_modules/minimatch": { + "version": "5.1.6", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.1.6.tgz", + "integrity": "sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==", + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/fill-range": { "version": "7.1.1", "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", @@ -443,6 +761,23 @@ "node": ">=8" } }, + "node_modules/finalhandler": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.3.1.tgz", + "integrity": "sha512-6BN9trH7bp3qvnrRyzsBz+g3lZxTNZTbVO2EV1CS0WIcDbawYVdYvGflME/9QP0h0pYlCDBCTjYa9nZzMDpyxQ==", + "dependencies": { + "debug": "2.6.9", + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "on-finished": "2.4.1", + "parseurl": "~1.3.3", + "statuses": "2.0.1", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, "node_modules/foreground-child": { "version": "3.3.0", "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.3.0.tgz", @@ -458,6 +793,22 @@ "url": "https://github.com/sponsors/isaacs" } }, + "node_modules/forwarded": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", + "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==", + "engines": { + "node": ">= 0.6" + } + }, "node_modules/fsevents": { "version": "2.3.3", "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz", @@ -480,6 +831,41 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/get-intrinsic": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.7.tgz", + "integrity": "sha512-VW6Pxhsrk0KAOqs3WEd0klDiF/+V7gQOpAvY1jVU/LHmaD/kQO4523aiJuikX/QAKYiW6x8Jh+RJej1almdtCA==", + "dependencies": { + "call-bind-apply-helpers": "^1.0.1", + "es-define-property": "^1.0.1", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", + "function-bind": "^1.1.2", + "get-proto": "^1.0.0", + "gopd": "^1.2.0", + "has-symbols": "^1.1.0", + "hasown": "^2.0.2", + "math-intrinsics": "^1.1.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-proto": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz", + "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==", + "dependencies": { + "dunder-proto": "^1.0.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + } + }, "node_modules/glob": { "version": "10.4.5", "resolved": "https://registry.npmjs.org/glob/-/glob-10.4.5.tgz", @@ -510,6 +896,36 @@ "node": ">=10.13.0" } }, + "node_modules/gopd": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz", + "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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==", + "engines": { + "node": ">=8" + } + }, + "node_modules/has-symbols": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz", + "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/hasown": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", @@ -521,6 +937,45 @@ "node": ">= 0.4" } }, + "node_modules/http-errors": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", + "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", + "dependencies": { + "depd": "2.0.0", + "inherits": "2.0.4", + "setprototypeof": "1.2.0", + "statuses": "2.0.1", + "toidentifier": "1.0.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "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==", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "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/ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", + "engines": { + "node": ">= 0.10" + } + }, "node_modules/is-binary-path": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", @@ -600,6 +1055,43 @@ "@pkgjs/parseargs": "^0.11.0" } }, + "node_modules/jake": { + "version": "10.9.2", + "resolved": "https://registry.npmjs.org/jake/-/jake-10.9.2.tgz", + "integrity": "sha512-2P4SQ0HrLQ+fw6llpLnOaGAvN2Zu6778SJMrCUwns4fOoG9ayrTiZk3VV8sCPkVZF8ab0zksVpS8FDY5pRCNBA==", + "dependencies": { + "async": "^3.2.3", + "chalk": "^4.0.2", + "filelist": "^1.0.4", + "minimatch": "^3.1.2" + }, + "bin": { + "jake": "bin/cli.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/jake/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==", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/jake/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, "node_modules/jiti": { "version": "1.21.7", "resolved": "https://registry.npmjs.org/jiti/-/jiti-1.21.7.tgz", @@ -629,6 +1121,30 @@ "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-10.4.3.tgz", "integrity": "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ==" }, + "node_modules/math-intrinsics": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz", + "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==", + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/merge-descriptors": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.3.tgz", + "integrity": "sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==", + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/merge2": { "version": "1.4.1", "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", @@ -637,6 +1153,14 @@ "node": ">= 8" } }, + "node_modules/methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==", + "engines": { + "node": ">= 0.6" + } + }, "node_modules/micromatch": { "version": "4.0.8", "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", @@ -649,6 +1173,36 @@ "node": ">=8.6" } }, + "node_modules/mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, "node_modules/minimatch": { "version": "9.0.5", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", @@ -671,6 +1225,11 @@ "node": ">=16 || 14 >=14.17" } }, + "node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, "node_modules/mz": { "version": "2.7.0", "resolved": "https://registry.npmjs.org/mz/-/mz-2.7.0.tgz", @@ -698,6 +1257,14 @@ "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" } }, + "node_modules/negotiator": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz", + "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==", + "engines": { + "node": ">= 0.6" + } + }, "node_modules/normalize-path": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", @@ -722,11 +1289,41 @@ "node": ">= 6" } }, + "node_modules/object-inspect": { + "version": "1.13.3", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.3.tgz", + "integrity": "sha512-kDCGIbxkDSXE3euJZZXzc6to7fCrKHNI/hSRQnRuQ+BWjFNzZwiFF8fj/6o2t2G9/jTj8PSIYTfCLelLZEeRpA==", + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/on-finished": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz", + "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==", + "dependencies": { + "ee-first": "1.1.1" + }, + "engines": { + "node": ">= 0.8" + } + }, "node_modules/package-json-from-dist": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/package-json-from-dist/-/package-json-from-dist-1.0.1.tgz", "integrity": "sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw==" }, + "node_modules/parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", + "engines": { + "node": ">= 0.8" + } + }, "node_modules/path-key": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", @@ -755,6 +1352,11 @@ "url": "https://github.com/sponsors/isaacs" } }, + "node_modules/path-to-regexp": { + "version": "0.1.12", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.12.tgz", + "integrity": "sha512-RA1GjUVMnvYFxuqovrEqZoxxW5NUZqbwKtYz/Tt7nXerk0LbLblQmrsgdeOxV5SFHf0UDggjS/bSeOZwt1pmEQ==" + }, "node_modules/picocolors": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", @@ -923,6 +1525,32 @@ "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz", "integrity": "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==" }, + "node_modules/proxy-addr": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", + "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", + "dependencies": { + "forwarded": "0.2.0", + "ipaddr.js": "1.9.1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/qs": { + "version": "6.13.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.13.0.tgz", + "integrity": "sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==", + "dependencies": { + "side-channel": "^1.0.6" + }, + "engines": { + "node": ">=0.6" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/queue-microtask": { "version": "1.2.3", "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", @@ -942,6 +1570,28 @@ } ] }, + "node_modules/range-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", + "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/raw-body": { + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.2.tgz", + "integrity": "sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==", + "dependencies": { + "bytes": "3.1.2", + "http-errors": "2.0.0", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, "node_modules/read-cache": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz", @@ -1011,6 +1661,85 @@ "queue-microtask": "^1.2.2" } }, + "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", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "node_modules/send": { + "version": "0.19.0", + "resolved": "https://registry.npmjs.org/send/-/send-0.19.0.tgz", + "integrity": "sha512-dW41u5VfLXu8SJh5bwRmyYUbAoSB3c9uQh6L8h/KtsFREPWpbX1lrljJo186Jc4nmci/sGUZ9a0a0J2zgfq2hw==", + "dependencies": { + "debug": "2.6.9", + "depd": "2.0.0", + "destroy": "1.2.0", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "2.0.0", + "mime": "1.6.0", + "ms": "2.1.3", + "on-finished": "2.4.1", + "range-parser": "~1.2.1", + "statuses": "2.0.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/send/node_modules/encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/send/node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" + }, + "node_modules/serve-static": { + "version": "1.16.2", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.16.2.tgz", + "integrity": "sha512-VqpjJZKadQB/PEbEwvFdO43Ax5dFBZ2UECszz8bQ7pi7wt//PWe1P6MN7eCnjsatYtBT6EuiClbjSWP2WrIoTw==", + "dependencies": { + "encodeurl": "~2.0.0", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", + "send": "0.19.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/setprototypeof": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", + "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==" + }, "node_modules/shebang-command": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", @@ -1030,6 +1759,74 @@ "node": ">=8" } }, + "node_modules/side-channel": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz", + "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.3", + "side-channel-list": "^1.0.0", + "side-channel-map": "^1.0.1", + "side-channel-weakmap": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-list": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz", + "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==", + "dependencies": { + "es-errors": "^1.3.0", + "object-inspect": "^1.13.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-map": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz", + "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/side-channel-weakmap": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz", + "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==", + "dependencies": { + "call-bound": "^1.0.2", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.5", + "object-inspect": "^1.13.3", + "side-channel-map": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/signal-exit": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-4.1.0.tgz", @@ -1049,6 +1846,14 @@ "node": ">=0.10.0" } }, + "node_modules/statuses": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", + "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==", + "engines": { + "node": ">= 0.8" + } + }, "node_modules/string-width": { "version": "5.1.2", "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz", @@ -1158,6 +1963,17 @@ "node": ">=16 || 14 >=14.17" } }, + "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==", + "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", @@ -1272,16 +2088,60 @@ "node": ">=8.0" } }, + "node_modules/toidentifier": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", + "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", + "engines": { + "node": ">=0.6" + } + }, "node_modules/ts-interface-checker": { "version": "0.1.13", "resolved": "https://registry.npmjs.org/ts-interface-checker/-/ts-interface-checker-0.1.13.tgz", "integrity": "sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==" }, + "node_modules/type-is": { + "version": "1.6.18", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", + "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", + "dependencies": { + "media-typer": "0.3.0", + "mime-types": "~2.1.24" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==", + "engines": { + "node": ">= 0.8" + } + }, "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/utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==", + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==", + "engines": { + "node": ">= 0.8" + } + }, "node_modules/which": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", @@ -1380,6 +2240,26 @@ "node": ">=8" } }, + "node_modules/ws": { + "version": "8.18.0", + "resolved": "https://registry.npmjs.org/ws/-/ws-8.18.0.tgz", + "integrity": "sha512-8VbfWfHLbbwu3+N6OKsOMpBdT4kXPDDB9cJk2bJ6mh9ucxdlnNvH1e+roYkKmN9Nxw2yjz7VzeO9oOz2zJ04Pw==", + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": ">=5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "utf-8-validate": { + "optional": true + } + } + }, "node_modules/yaml": { "version": "2.6.1", "resolved": "https://registry.npmjs.org/yaml/-/yaml-2.6.1.tgz", diff --git a/package.json b/package.json index 8ae4fe7..9ebf517 100644 --- a/package.json +++ b/package.json @@ -25,7 +25,9 @@ "didyoumean": "^1.2.2", "dlv": "^1.1.3", "eastasianwidth": "^0.2.0", + "ejs": "^3.1.10", "emoji-regex": "^9.2.2", + "express": "^4.21.2", "fast-glob": "^3.3.2", "fastq": "^1.18.0", "fill-range": "^7.1.1", @@ -94,6 +96,7 @@ "which": "^2.0.2", "wrap-ansi": "^8.1.0", "wrap-ansi-cjs": "^7.0.0", + "ws": "^8.18.0", "yaml": "^2.6.1" }, "scripts": { diff --git a/src/config.js b/src/config.js index 5fb19a6..7803625 100644 --- a/src/config.js +++ b/src/config.js @@ -1,7 +1,7 @@ const config = { - stickerPacksDir: "./stickerpacks", - outDir: "./", + stickerPacksDir: "stickerpacks", + outDir: "", homeserverUrl: "https://synapse.wah.su/__thumbnail/" -} +}; module.exports = config \ No newline at end of file diff --git a/src/hotreload.js b/src/hotreload.js new file mode 100644 index 0000000..f9de754 --- /dev/null +++ b/src/hotreload.js @@ -0,0 +1,11 @@ +let webSocket = new WebSocket(`ws://${window.location.hostname}:3001`); +webSocket.onmessage = function(e) { + if (e.data == "RELOAD") { + console.log("Reloading page after build") + location.reload() + } else if (e.data == "CONNECTED") { + console.log("Connected to server") + } else { + console.warn(`unknown data received: ${e}`) + } +}; \ No newline at end of file diff --git a/src/index.js b/src/index.js index ca5f339..e8fe6ac 100644 --- a/src/index.js +++ b/src/index.js @@ -1,44 +1,120 @@ const config = require("./config"); const fs = require("fs"); +const path = require("path"); +const { log, CreateImageURL, CreatePackDescription } = require("./utils"); +let ejs = require("ejs"); -const _CreatePackPage = require("./templates/pack"); -const _CreatePacksIndex = require("./templates/index"); - -let PackIndex = null +let PackIndex = null; let Packs = []; -const dirents = fs.readdirSync(config.stickerPacksDir, { withFileTypes: true }); +const isDev = process.env.DEVMODE || false; +log("INFO", `DEV MODE ENABLED: ${isDev}`); + +const InpPath = path + .join(__dirname, "../", config.stickerPacksDir, "./") + .trim(); +const OutPath = path.join(__dirname, "../", config.outDir, "./").trim(); +const ParPath = path.join(__dirname, "../", "./").trim(); +log("INFO", `Sticker sets directory: ${InpPath}`); +log("INFO", `Output directory: ${OutPath}`); +log("INFO", `Working directory: ${ParPath}`); + +const dirents = fs.readdirSync(InpPath, { withFileTypes: true }); const files = dirents - .filter(dirent => dirent.isFile()) - .filter(dirent => (dirent.name.endsWith(".json") && dirent.name != "index.json")) - .map(dirent => dirent.name); + .filter((dirent) => dirent.isFile()) + .filter( + (dirent) => dirent.name.endsWith(".json") && dirent.name != "index.json" + ) + .map((dirent) => dirent.name); if (files.length == 0) { - console.error("[ERROR] NO Sticker Packs Found!"); - process.exit(1); + log("error", "No sticker sets found!"); + process.exit(1); } -console.log("[INFO] Found " + files.length + " sticker packs"); +log("INFO", `Found: ${files.length} sticker sets`); PackIndex = { - homeserver_url: config.homeserverUrl, - packs: files -} -fs.writeFileSync(config.stickerPacksDir + "/index.json", JSON.stringify(PackIndex)); + homeserver_url: config.homeserverUrl, + packs: files, +}; +fs.writeFileSync( + config.stickerPacksDir + "/index.json", + JSON.stringify(PackIndex) +); +log("INFO", `Updated "index.json" in sticker sets directory`, true); -if (!fs.existsSync(config.outDir)) fs.mkdirSync(config.outDir); +if (OutPath != ParPath) { + if (!fs.existsSync(OutPath)) fs.mkdirSync(OutPath); + fs.cpSync(`${ParPath}static`, `${OutPath}static`, { recursive: true }); + log("INFO", `Copied static directory to output directory`); +} PackIndex.packs.forEach((pack) => { - const packFile = JSON.parse(fs.readFileSync(config.stickerPacksDir + "/" + pack)); - if (!fs.existsSync(config.outDir + "/" + packFile.id)) fs.mkdirSync(config.outDir + "/" + packFile.id); - fs.writeFileSync(config.outDir + "/" + packFile.id + "/index.html", _CreatePackPage(PackIndex, packFile)); - Packs.push({ - id: packFile.id, - name: packFile.title, - image: packFile.stickers[0].id, - author: (packFile.hasOwnProperty("author") && packFile.author) ? packFile.author.name : null, - rating: packFile.hasOwnProperty("rating") ? packFile.rating : null, - stickers: packFile.stickers.length - }) - console.log("preview for " + packFile.id + " created"); -}) -fs.writeFileSync(config.outDir + "/index.html", _CreatePacksIndex(PackIndex, Packs)); -console.log("Generation complete"); + const packFile = JSON.parse(fs.readFileSync(`${InpPath}${pack}`)); + const Template = fs.readFileSync( + path.join(ParPath, "src/templates/Base.ejs") + ); + const html = ejs.render( + Template.toString(), + { + title: packFile.title, + description: CreatePackDescription(packFile), + image: { + url: CreateImageURL(config.homeserverUrl, packFile.stickers[0].id), + mimetype: packFile.stickers[0].info.mimetype, + w: packFile.stickers[0].info.w, + h: packFile.stickers[0].info.h, + alt: packFile.stickers[0].info.alt, + }, + path: config.outDir ? `/${config.outDir}` : "", + isDev, + stickerset: packFile, + page: "stickerset", + homeserverUrl: config.homeserverUrl, + }, + { root: path.join(ParPath, "src/templates") } + ); + if (!fs.existsSync(`${OutPath}${packFile.id}`)) fs.mkdirSync(`${OutPath}${packFile.id}`); + fs.writeFileSync(`${OutPath}${packFile.id}/index.html`, html); + Packs.push({ + id: packFile.id, + name: packFile.title, + image: packFile.stickers[0].id, + author: + packFile.hasOwnProperty("author") && packFile.author + ? packFile.author.name + : null, + rating: packFile.hasOwnProperty("rating") ? packFile.rating : null, + stickers: packFile.stickers.length, + }); + log( + "INFO", + `Created preview for sticker set: ${packFile.title} (${packFile.id})` + ); +}); + +const indexTemplate = fs.readFileSync( + path.join(ParPath, "src/templates/Base.ejs") +); +const html = ejs.render( + indexTemplate.toString(), + { + title: "TG -> Matrix Stickers Index", + description: `Available ${PackIndex.packs.length} sticker packs`, + image: { + url: "./static/images/sticker.png", + mimetype: "image/png", + w: "96", + h: "96", + alt: "", + }, + path: config.outDir ? `/${config.outDir}` : "", + isDev, + stickerset: null, + page: "index", + packs: Packs, + homeserverUrl: config.homeserverUrl, + }, + { root: path.join(ParPath, "src/templates") } +); +fs.writeFileSync(`${OutPath}index.html`, html); +log("INFO", "Generation complete"); diff --git a/src/templates/Base.ejs b/src/templates/Base.ejs new file mode 100644 index 0000000..c148b2e --- /dev/null +++ b/src/templates/Base.ejs @@ -0,0 +1,100 @@ + + +<%- include('/components/Head', {title, description, image, path, isDev}); %> + + <%- include("/components/Header", {path, stickerset}) %> +
+
+
+ <% if (page=="index" ) { %> +
+ <% packs.forEach(function(pack){ %> + <%- include('/components/Link/Stickerset', {path, homeserverUrl, pack}); %> + <% }); %> +
+ <%- include('/components/Footer'); %> + <% }; %> + <% if (page=="stickerset" ) { %> +
+
+ <%- include('/components/Pack/Description', {homeserverUrl, stickerset}); %> + <%- include('/components/Pack/Links', {stickerset}); %> + + +
+
+ <%- include('/components/Pack/Preview', {homeserverUrl, stickerset}); %> +
+
+ <%- include('/components/Footer'); %> +
+
+ <% }; %> +
+ + + <% if (page=="stickerset" ) { %> + + <% }; %> + \ No newline at end of file diff --git a/src/templates/components/Footer.ejs b/src/templates/components/Footer.ejs new file mode 100644 index 0000000..6688028 --- /dev/null +++ b/src/templates/components/Footer.ejs @@ -0,0 +1,18 @@ +
+ + by + @radiquum +
+

Find us on:

+ github + website +
+
\ No newline at end of file diff --git a/src/templates/components/Head.ejs b/src/templates/components/Head.ejs new file mode 100644 index 0000000..5bfebdd --- /dev/null +++ b/src/templates/components/Head.ejs @@ -0,0 +1,20 @@ + + + + <%- title %> + + + + + + + + + + + <% if (isDev) { %> + + + + <% }; %> + \ No newline at end of file diff --git a/src/templates/components/Header.ejs b/src/templates/components/Header.ejs new file mode 100644 index 0000000..cfbe8d2 --- /dev/null +++ b/src/templates/components/Header.ejs @@ -0,0 +1,15 @@ +<% include('/functions') %> +
+
+ index page + <% if (stickerset) { %> +
+ +

<%- stickerset.title %>

+
+ <% }; %> + +
+
\ No newline at end of file diff --git a/src/templates/components/Link/AddLink.ejs b/src/templates/components/Link/AddLink.ejs new file mode 100644 index 0000000..abda97e --- /dev/null +++ b/src/templates/components/Link/AddLink.ejs @@ -0,0 +1,4 @@ + + +

<%- text %>

+
\ No newline at end of file diff --git a/src/templates/components/Link/Stickerset.ejs b/src/templates/components/Link/Stickerset.ejs new file mode 100644 index 0000000..26ba4d7 --- /dev/null +++ b/src/templates/components/Link/Stickerset.ejs @@ -0,0 +1,29 @@ +<% include('/functions') %> + +
+ + +
+
+

<%- pack.name %>

+

<%- pack.author %>

+

<%- pack.rating %> | <%- pack.stickers %> stick

+
+
\ No newline at end of file diff --git a/src/templates/components/Pack/Description.ejs b/src/templates/components/Pack/Description.ejs new file mode 100644 index 0000000..f4c5f89 --- /dev/null +++ b/src/templates/components/Pack/Description.ejs @@ -0,0 +1,37 @@ +<% include('/functions') %> +
+
+ + +
+
+

<%- stickerset.title %>

+ <% if (stickerset.hasOwnProperty("author") && stickerset.author.hasOwnProperty("name") && stickerset.author.name) { %> + <% if (stickerset.author.hasOwnProperty("url") && stickerset.author.url) { %> + <%- stickerset.author.name %> + <% } else { %> +

<%- stickerset.author.name %>

+ <% }; %> + <% }; %> + <% if (stickerset.hasOwnProperty("rating") && stickerset.rating) { %> +

<%- stickerset.rating %>

+ <% }; %> +
+
\ No newline at end of file diff --git a/src/templates/components/Pack/Links.ejs b/src/templates/components/Pack/Links.ejs new file mode 100644 index 0000000..bdebca9 --- /dev/null +++ b/src/templates/components/Pack/Links.ejs @@ -0,0 +1,5 @@ +<%- include ("/components/Link/AddLink", {name: "telegram", text: "Telegram", image: "telegram.png", link: "https://t.me/addstickers/" + stickerset.id, color: "#2f7ca3"} ) %> +<% if (stickerset.hasOwnProperty("room_id") && stickerset.room_id) { %> + <%- include ("/components/Link/AddLink", {name: "cinny.png", text: "Cinny", image: "cinny.png", link: "https://matrix.to/#/" + stickerset.room_id, color: "#373737"} ) %> + <%- include ("/components/Link/AddLink", {name: "fluffychat.png", text: "FluffyChat", image: "fluffychat.png", link: "https://matrix.to/#/" + stickerset.room_id, color: "#282443"} ) %> +<% }; %> \ No newline at end of file diff --git a/src/templates/components/Pack/Preview.ejs b/src/templates/components/Pack/Preview.ejs new file mode 100644 index 0000000..4793bc1 --- /dev/null +++ b/src/templates/components/Pack/Preview.ejs @@ -0,0 +1,30 @@ +<% include('/functions') %> +
+

<%- stickerset.stickers.length %> stickers

+
+ <% stickerset.stickers.forEach(function(sticker){ %> +
+ + +
+ <% }); %> +
+ +
\ No newline at end of file diff --git a/src/templates/components/packCard.js b/src/templates/components/packCard.js deleted file mode 100644 index f3bba4c..0000000 --- a/src/templates/components/packCard.js +++ /dev/null @@ -1,71 +0,0 @@ -const { CreateImageURL } = require("../../utils"); - -function _packName(pack) { - let string = ""; - string += `

${pack.title}

`; - - if (pack.hasOwnProperty("author") && pack.author) { - let author_string = `

by: ${pack.author.name}

`; - if (pack.author.url) { - author_string = `

by: ${pack.author.name}

`; - } - string += author_string; - } - - if (pack.hasOwnProperty("rating") && pack.rating) { - switch (pack.rating.toLowerCase()) { - case "safe": - string += `

safe

`; - break; - case "questionable": - string += `

questionable

`; - break; - case "explicit": - string += `

explicit

`; - break; - default: - break; - } - } - - return string; -} - -function _packCard(index, pack) { - return ` -
-
-
- - Loading... -
- -
-
- ${_packName(pack)} -
-
-`; -} - -module.exports = _packCard; diff --git a/src/templates/components/packHead.js b/src/templates/components/packHead.js deleted file mode 100644 index 0ceb6fb..0000000 --- a/src/templates/components/packHead.js +++ /dev/null @@ -1,20 +0,0 @@ -const {CreateImageURL, CreatePackDescription} = require("../../utils"); - -function _CreateHead(index, pack) { - return ` - - -${pack.title} - - - - - - - - - - -`} - -module.exports = _CreateHead \ No newline at end of file diff --git a/src/templates/components/packLinks.js b/src/templates/components/packLinks.js deleted file mode 100644 index 7d4c7ba..0000000 --- a/src/templates/components/packLinks.js +++ /dev/null @@ -1,26 +0,0 @@ -function PackLinks(pack) { - return ` - - -

Telegram

-
- - ${(pack.hasOwnProperty("room_id") && pack.room_id) ? ( - ` - - -

FluffyChat

-
- - -

Cinny

-
- ` - ) - : ""} -`} - -module.exports = PackLinks \ No newline at end of file diff --git a/src/templates/components/packPreview.js b/src/templates/components/packPreview.js deleted file mode 100644 index 14072d3..0000000 --- a/src/templates/components/packPreview.js +++ /dev/null @@ -1,48 +0,0 @@ -const { CreateImageURL } = require("../../utils"); - -function _addSticker(index, sticker) { - return ` -
-
- - Loading... -
- -
-`}; - -function _packPreview(index, pack) { - - let stickers = []; - pack.stickers.forEach((sticker) => stickers.push(_addSticker(index, sticker))); - - return ` -
-

Pack Preview (${pack.stickers.length} stickers)

-
- ${stickers.join("\n")} -
-
-`}; - -module.exports = _packPreview; diff --git a/src/templates/functions.ejs b/src/templates/functions.ejs new file mode 100644 index 0000000..4719398 --- /dev/null +++ b/src/templates/functions.ejs @@ -0,0 +1,19 @@ +<% CreateImageURL=function(homeserverUrl,id) { return `${homeserverUrl}${id.slice(0,2)}/${id.slice(2,4)}/${id.slice(4)}`; } %> +<% +PackRatingClass=function(rating) { + switch (rating.toLowerCase()) { + case "safe": + return "text-green-400" + break + case "questionable": + return "text-yellow-500" + break + case "explicit": + return "text-red-500" + break + default: + return "hidden" + break + } +} +%> \ No newline at end of file diff --git a/src/templates/index.js b/src/templates/index.js deleted file mode 100644 index f287867..0000000 --- a/src/templates/index.js +++ /dev/null @@ -1,97 +0,0 @@ -const { CreateImageURL } = require("../utils"); - -function _PackLink(index, pack) { - - let packRating = ""; - - if (pack.rating) { - if (pack.rating.toLowerCase() === "safe") { - packRating = `S`; - } else if (pack.rating.toLowerCase() === "questionable") { - packRating = `Q`; - } else if (pack.rating.toLowerCase() === "explicit") { - packRating = `E`; - } - } - - return ` - - -
- -
- - Loading... -
- -
- -
-

${pack.name}

- ${pack.author ? `

${pack.author}

` : ""} -

${pack.stickers} stickers ${pack.rating ? "| " + packRating : ""}

-
- -
-` -} - -function _CreatePacksIndex(index, packs) { - - let packLinks = []; - packs.forEach((packLink) => packLinks.push(_PackLink(index, packLink))); - - return ` - - - - - - TG -> Matrix Stickers Index - - - - - - - - - - - - -
-
- -
-
- ${packLinks.join("\n")} -
-
- - - -` -} - -module.exports = _CreatePacksIndex \ No newline at end of file diff --git a/src/templates/pack.js b/src/templates/pack.js deleted file mode 100644 index 82f92ef..0000000 --- a/src/templates/pack.js +++ /dev/null @@ -1,66 +0,0 @@ -const PackHead = require("./components/packHead"); -const PackCard = require("./components/packCard"); -const PackLinks = require("./components/packLinks"); -const PackPreview = require("./components/packPreview"); - -function _CreatePackPage(index, pack) { - return ` - - - - - ${PackHead(index, pack)} - - - - -
-
- -
- ${PackCard(index, pack)} -
- ${PackLinks(pack)} -
- ${PackPreview(index, pack)} -
- - - - - - - - - ` -} - -module.exports = _CreatePackPage \ No newline at end of file diff --git a/src/utils.js b/src/utils.js index 86cd5e3..68cc965 100644 --- a/src/utils.js +++ b/src/utils.js @@ -1,8 +1,6 @@ -function CreateImageURL(index, id) { - return `${index.homeserver_url}${id.slice(0, 2)}/${id.slice(2, 4)}/${id.slice( - 4 - )}`; -} +CreateImageURL = function (homeserverUrl, id) { + return `${homeserverUrl}${id.slice(0, 2)}/${id.slice(2, 4)}/${id.slice(4)}`; +}; function CreatePackDescription(pack) { let description = []; @@ -36,4 +34,34 @@ function CreatePackDescription(pack) { return description.join(" | "); } -module.exports = {CreateImageURL, CreatePackDescription}; +function log( + level = "INFO" | "ERROR" | "WARN" | "LOG", + message, + connected = false +) { + const date = new Date(); + const time = date.toLocaleTimeString(); + if (connected) { + message = `↳${message}`; + } + switch (level.toUpperCase()) { + case "INFO": + console.info(`${time}:${level} - ${message}`); + break; + case "ERROR": + console.error(`${time}:${level} - ${message}`); + break; + case "WARN": + console.warn(`${time}:${level} - ${message}`); + break; + default: + console.log(`${time}:LOG - ${message}`); + break; + } +} + +module.exports = { + log, + CreatePackDescription, + CreateImageURL +}; diff --git a/src/watch.js b/src/watch.js index 172462d..2418555 100644 --- a/src/watch.js +++ b/src/watch.js @@ -1,36 +1,136 @@ const chokidar = require("chokidar"); const exec = require("child_process"); +const express = require("express"); +const path = require("path"); +const WebSocket = require("ws"); +const fs = require("fs"); +const config = require("./config"); +const { log } = require("./utils"); + +let triggered = 0; +const delay = 1000; +let SIGINTCount = 0; +let WSclients = []; + +const InpPath = path + .join(__dirname, "../", config.stickerPacksDir, "./") + .trim(); +const OutPath = path.join(__dirname, "../", config.outDir, "./").trim(); +const ParPath = path.join(__dirname, "../", "./").trim(); +log("INFO", `Sticker sets directory: ${InpPath}`); +log("INFO", `Output directory: ${OutPath}`); +log("INFO", `Working directory: ${ParPath}`); function onChange() { + if (triggered != 0 && Date.now() - triggered < delay) { + log("WARN", `Rebuild was triggered less than a ${delay}ms ago!`, true); + return; + } + triggered = Date.now(); + + process.env.DEVMODE = true; exec.exec("npm run build", (error, stdout, stderr) => { if (error) { - console.error(`error: ${error.message}`); + log("ERROR", error.message); return; } if (stderr) { - console.error(`stderr: ${stderr}`); + log("ERROR", stderr); return; } - console.log(`stdout: ${stdout}`); + log("INFO", stdout); + if (WSclients.length > 0) { + log("INFO", "Reloading web page..."); + WSclients.forEach((ws) => ws.send("RELOAD")); + } }); } -const watcher = chokidar.watch("./src/templates", { - ignored: (path, stats) => - stats?.isFile() && !(path.endsWith(".js") || path.endsWith(".json")), - // atomic: true, +function onExit() { + let folders = null; + + if (fs.existsSync(`${InpPath}/index.json`)) { + folders = []; + JSON.parse(fs.readFileSync(`${InpPath}/index.json`))["packs"].map((pack) => + folders.push(pack.replace(".json", "")) + ); + + folders.forEach((folder) => { + if (fs.existsSync(`${OutPath}${folder}`)) + fs.rmdirSync(`${OutPath}${folder}`, { recursive: true }); + log("INFO", `Deleted generated folder: "${folder}"`); + }); + } else { + log("WARN", `no "index.json" found, forgot to run build?`); + } + + if (fs.existsSync(`${OutPath}index.html`)) fs.rmSync(`${OutPath}index.html`); + log("INFO", `Deleted "index.html" file`); + + if (fs.existsSync(OutPath) && OutPath != ParPath) { + fs.rmdirSync(`${OutPath}`, { recursive: true }); + log("INFO", `Deleted output folder`); + } + + process.exit(0); +} + +const watcher = chokidar.watch(["./src", "./stickerpacks", "./static"], { + ignored: (filePath, stats) => filePath.endsWith("index.json"), + atomic: true, awaitWriteFinish: true, persistent: true, }); -watcher.on("add", (path) => { - console.log(`File ${path} has been added, rebuilding...`); - onChange(); -}); -watcher.on("change", (path) => { - console.log(`File ${path} has been changed, rebuilding...`); - onChange(); -}); -watcher.on("unlink", (path) => { - console.log(`File ${path} has been removed, rebuilding...`); - onChange(); -}); + +function startServerWithRebuild() { + const app = express(); + const folder = path.join(__dirname, ".."); + const wss = new WebSocket.Server({ port: 3001 }); + + + wss.on("connection", (ws) => { + WSclients.push(ws); + log("INFO", `Client ${WSclients.length} connected`) + ws.send("CONNECTED"); + }); + + + process.on("SIGINT", () => { + SIGINTCount += 1; + if (WSclients.length > 0) { + async function _closeWS() { + WSclients.forEach(async (ws) => await ws.close()) + } + _closeWS(); + } + if (SIGINTCount == 1) { + log("LOG", "Gracefully shutdown and cleanup..."); + onExit(); + } else if (SIGINTCount >= 3) { + log("LOG", "Received 3+ SIGINT signals. Force exit..."); + process.exit(0); + } + }); + + app.use(express.static(folder)); + app.listen(3000, () => { + log("INFO", `Serving files from folder ${folder}`); + log("INFO", "Express server is running on port 3000"); + + watcher + .on("add", (path) => { + log("INFO", `File ${path} has been added, rebuilding...`); + onChange(); + }) + .on("change", (path) => { + log("INFO", `File ${path} has been changed, rebuilding...`); + onChange(); + }) + .on("unlink", (path) => { + log("INFO", `File ${path} has been removed, rebuilding...`); + onChange(); + }); + }); +} + +startServerWithRebuild(); diff --git a/static/OpenPopUp.js b/static/OpenPopUp.js deleted file mode 100644 index 6fbdfb4..0000000 --- a/static/OpenPopUp.js +++ /dev/null @@ -1,13 +0,0 @@ -const ElementInstructionOV = document.getElementById( - "preview_sticker_pack_add_to_element_overlay" -); -const ElementInstruction = document.getElementById( - "preview_sticker_pack_add_to_element" -); - -function toggleElementInstruction() { - ElementInstructionOV.classList.toggle("hidden"); - ElementInstruction.classList.toggle("hidden"); - - ElementInstruction.classList.toggle("flex"); -} diff --git a/static/RenderImages.js b/static/RenderImages.js index bc6b287..b26f0e8 100644 --- a/static/RenderImages.js +++ b/static/RenderImages.js @@ -1,22 +1,26 @@ const images = document.querySelectorAll("[data-image-id]"); +images.forEach((image, i) => { + if (i < 4) { + image.setAttribute("loading", "eager"); + } -images.forEach((image) => { + const spinner = document.querySelector( + `[data-spinner-id="${image.getAttribute("data-image-id")}"]` + ); - const spinner = document.querySelector(`[data-spinner-id="${image.getAttribute("data-image-id")}"]`) - console.log(image) - console.log(spinner) + if (image.height > 0 && image.complete) { + image.classList.remove("invisible"); + spinner.classList.add("invisible"); + return; + } else { + image.classList.add("invisible"); + spinner.classList.remove("invisible"); + } - if (image.height > 0) { - image.classList.remove("hidden"); - spinner.classList.add("hidden"); - return - } - - image.addEventListener("load", () => { - console.log("image " + image.getAttribute("data-image-id") + " loaded"); - image.classList.remove("hidden"); - spinner.classList.add("hidden"); - image.removeEventListener("load", this); - }); - -}); \ No newline at end of file + image.addEventListener("load", () => { + console.log("image " + image.getAttribute("data-image-id") + " loaded"); + image.classList.remove("invisible"); + spinner.classList.add("invisible"); + image.removeEventListener("load", this); + }); +}); diff --git a/static/images/captive_portal_24dp_E8EAED_FILL0_wght400_GRAD0_opsz24.png b/static/images/captive_portal_24dp_E8EAED_FILL0_wght400_GRAD0_opsz24.png new file mode 100644 index 0000000..0908872 Binary files /dev/null and b/static/images/captive_portal_24dp_E8EAED_FILL0_wght400_GRAD0_opsz24.png differ diff --git a/static/images/github-mark-white.png b/static/images/github-mark-white.png new file mode 100644 index 0000000..50b8175 Binary files /dev/null and b/static/images/github-mark-white.png differ diff --git a/static/images/logo.svg b/static/images/logo.svg new file mode 100644 index 0000000..0c927dc --- /dev/null +++ b/static/images/logo.svg @@ -0,0 +1,52 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/static/tailwind.css b/static/tailwind.css index 8ba6f24..b54ece5 100644 --- a/static/tailwind.css +++ b/static/tailwind.css @@ -588,32 +588,32 @@ video { } } -.sr-only { - position: absolute; - width: 1px; - height: 1px; - padding: 0; - margin: -1px; - overflow: hidden; - clip: rect(0, 0, 0, 0); - white-space: nowrap; - border-width: 0; +.invisible { + visibility: hidden; } .fixed { position: fixed; } +.sticky { + position: sticky; +} + .inset-0 { inset: 0px; } -.left-1\/2 { - left: 50%; +.left-0 { + left: 0px; } -.top-8 { - top: 2rem; +.right-0 { + right: 0px; +} + +.top-0 { + top: 0px; } .-z-10 { @@ -628,15 +628,26 @@ video { z-index: 10; } -.z-20 { - z-index: 20; -} - .mx-auto { margin-left: auto; margin-right: auto; } +.mb-4 { + margin-bottom: 1rem; +} + +.line-clamp-1 { + overflow: hidden; + display: -webkit-box; + -webkit-box-orient: vertical; + -webkit-line-clamp: 1; +} + +.block { + display: block; +} + .inline { display: inline; } @@ -653,16 +664,25 @@ video { display: none; } -.h-16 { - height: 4rem; +.aspect-square { + aspect-ratio: 1 / 1; } -.h-24 { - height: 6rem; +.h-6 { + height: 1.5rem; } -.h-48 { - height: 12rem; +.h-fit { + height: -moz-fit-content; + height: fit-content; +} + +.h-full { + height: 100%; +} + +.min-h-16 { + min-height: 4rem; } .min-h-screen { @@ -681,20 +701,16 @@ video { width: 6rem; } -.w-48 { - width: 12rem; +.w-8 { + width: 2rem; } .w-full { width: 100%; } -.flex-grow { - flex-grow: 1; -} - -.-translate-x-1\/2 { - --tw-translate-x: -50%; +.translate-y-\[var\(--tw-translate-y\)\] { + --tw-translate-y: var(--tw-translate-y); transform: translate(var(--tw-translate-x), var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y)); } @@ -708,8 +724,16 @@ video { animation: spin 1s linear infinite; } -.flex-row { - flex-direction: row; +.grid-cols-1 { + grid-template-columns: repeat(1, minmax(0, 1fr)); +} + +.grid-cols-3 { + grid-template-columns: repeat(3, minmax(0, 1fr)); +} + +.grid-rows-1 { + grid-template-rows: repeat(1, minmax(0, 1fr)); } .flex-col { @@ -728,6 +752,10 @@ video { justify-content: center; } +.justify-between { + justify-content: space-between; +} + .gap-2 { gap: 0.5rem; } @@ -736,69 +764,52 @@ video { gap: 1rem; } -.gap-1 { - gap: 0.25rem; -} - .overflow-x-hidden { overflow-x: hidden; } +.whitespace-pre { + white-space: pre; +} + +.rounded-full { + border-radius: 9999px; +} + .rounded-lg { border-radius: 0.5rem; } -.border-2 { - border-width: 2px; +.rounded-md { + border-radius: 0.375rem; } -.border-solid { - border-style: solid; +.rounded-b-lg { + border-bottom-right-radius: 0.5rem; + border-bottom-left-radius: 0.5rem; } -.border-black { +.border-b { + border-bottom-width: 1px; +} + +.border-slate-400 { --tw-border-opacity: 1; - border-color: rgb(0 0 0 / var(--tw-border-opacity, 1)); + border-color: rgb(148 163 184 / var(--tw-border-opacity, 1)); } -.bg-\[\#259d7b\] { +.bg-\[\#1d1f3d\] { --tw-bg-opacity: 1; - background-color: rgb(37 157 123 / var(--tw-bg-opacity, 1)); + background-color: rgb(29 31 61 / var(--tw-bg-opacity, 1)); } -.bg-\[\#282443\] { - --tw-bg-opacity: 1; - background-color: rgb(40 36 67 / var(--tw-bg-opacity, 1)); +.bg-\[var\(--bg-color\)\] { + background-color: var(--bg-color); } -.bg-\[\#2f7ca3\] { +.bg-slate-800 { --tw-bg-opacity: 1; - background-color: rgb(47 124 163 / var(--tw-bg-opacity, 1)); -} - -.bg-\[\#373737\] { - --tw-bg-opacity: 1; - background-color: rgb(55 55 55 / var(--tw-bg-opacity, 1)); -} - -.bg-black { - --tw-bg-opacity: 1; - background-color: rgb(0 0 0 / var(--tw-bg-opacity, 1)); -} - -.bg-red-600 { - --tw-bg-opacity: 1; - background-color: rgb(220 38 38 / var(--tw-bg-opacity, 1)); -} - -.bg-stone-800 { - --tw-bg-opacity: 1; - background-color: rgb(41 37 36 / var(--tw-bg-opacity, 1)); -} - -.bg-stone-900 { - --tw-bg-opacity: 1; - background-color: rgb(28 25 23 / var(--tw-bg-opacity, 1)); + background-color: rgb(30 41 59 / var(--tw-bg-opacity, 1)); } .bg-gradient-to-b { @@ -848,19 +859,19 @@ video { padding-bottom: 1rem; } -.py-8 { - padding-top: 2rem; - padding-bottom: 2rem; -} - .text-2xl { font-size: 1.5rem; line-height: 2rem; } -.text-4xl { - font-size: 2.25rem; - line-height: 2.5rem; +.text-lg { + font-size: 1.125rem; + line-height: 1.75rem; +} + +.text-sm { + font-size: 0.875rem; + line-height: 1.25rem; } .text-xl { @@ -868,6 +879,15 @@ video { line-height: 1.75rem; } +.text-xs { + font-size: 0.75rem; + line-height: 1rem; +} + +.font-bold { + font-weight: 700; +} + .text-gray-200 { --tw-text-opacity: 1; color: rgb(229 231 235 / var(--tw-text-opacity, 1)); @@ -898,46 +918,22 @@ video { color: rgb(234 179 8 / var(--tw-text-opacity, 1)); } -.text-zinc-100 { - --tw-text-opacity: 1; - color: rgb(244 244 245 / var(--tw-text-opacity, 1)); -} - -.text-slate-200 { - --tw-text-opacity: 1; - color: rgb(226 232 240 / var(--tw-text-opacity, 1)); -} - .underline { text-decoration-line: underline; } -.opacity-40 { - opacity: 0.4; -} - -.shadow-lg { - --tw-shadow: 0 10px 15px -3px rgb(0 0 0 / 0.1), 0 4px 6px -4px rgb(0 0 0 / 0.1); - --tw-shadow-colored: 0 10px 15px -3px var(--tw-shadow-color), 0 4px 6px -4px var(--tw-shadow-color); - box-shadow: var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow); -} - .transition-transform { transition-property: transform; transition-timing-function: cubic-bezier(0.4, 0, 0.2, 1); transition-duration: 150ms; } -.\[grid-template-columns\:100\%\] { - grid-template-columns: 100%; +.\[grid-column\:1\] { + grid-column: 1; } -.\[grid-template-columns\:repeat\(auto-fill\2c 96px\)\] { - grid-template-columns: repeat(auto-fill,96px); -} - -.\[grid-template-columns\:repeat\(auto-fill\2c minmax\(128px\2c 372px\)\)\] { - grid-template-columns: repeat(auto-fill,minmax(128px,372px)); +.\[grid-row\:1\] { + grid-row: 1; } .tiledBackground { @@ -956,33 +952,107 @@ video { transform: translate(var(--tw-translate-x), var(--tw-translate-y)) rotate(var(--tw-rotate)) skewX(var(--tw-skew-x)) skewY(var(--tw-skew-y)) scaleX(var(--tw-scale-x)) scaleY(var(--tw-scale-y)); } +.hover\:underline:hover { + text-decoration-line: underline; +} + +@media (min-width: 640px) { + .sm\:flex { + display: flex; + } + + .sm\:h-10 { + height: 2.5rem; + } + + .sm\:w-16 { + width: 4rem; + } + + .sm\:w-32 { + width: 8rem; + } + + .sm\:grid-cols-4 { + grid-template-columns: repeat(4, minmax(0, 1fr)); + } + + .sm\:gap-4 { + gap: 1rem; + } + + .sm\:px-8 { + padding-left: 2rem; + padding-right: 2rem; + } + + .sm\:text-2xl { + font-size: 1.5rem; + line-height: 2rem; + } + + .sm\:text-base { + font-size: 1rem; + line-height: 1.5rem; + } + + .sm\:text-lg { + font-size: 1.125rem; + line-height: 1.75rem; + } + + .sm\:text-xl { + font-size: 1.25rem; + line-height: 1.75rem; + } +} + @media (min-width: 768px) { - .md\:top-32 { - top: 8rem; + .md\:block { + display: block; } - .md\:w-\[768px\] { - width: 768px; + .md\:text-2xl { + font-size: 1.5rem; + line-height: 2rem; + } +} + +@media (min-width: 1024px) { + .lg\:block { + display: block; } - .md\:max-w-\[768px\] { - max-width: 768px; + .lg\:hidden { + display: none; } - .md\:flex-row { - flex-direction: row; + .lg\:w-\[49\%\] { + width: 49%; } - .md\:whitespace-pre { - white-space: pre; + .lg\:grid-cols-2 { + grid-template-columns: repeat(2, minmax(0, 1fr)); } - .md\:\[grid-template-columns\:repeat\(auto-fill\2c 384px\)\] { - grid-template-columns: repeat(auto-fill,384px); + .lg\:grid-cols-3 { + grid-template-columns: repeat(3, minmax(0, 1fr)); + } +} + +@media (min-width: 1280px) { + .xl\:grid-cols-4 { + grid-template-columns: repeat(4, minmax(0, 1fr)); + } +} + +@media (min-width: 1536px) { + .\32xl\:grid-cols-3 { + grid-template-columns: repeat(3, minmax(0, 1fr)); } - .md\:\[grid-template-columns\:repeat\(auto-fill\2c minmax\(380px\2c 500px\)\)\] { - grid-template-columns: repeat(auto-fill,minmax(380px,500px)); + .\32xl\:grid-cols-5 { + grid-template-columns: repeat(5, minmax(0, 1fr)); } } diff --git a/tailwind.config.js b/tailwind.config.js index e00955b..45d37f2 100644 --- a/tailwind.config.js +++ b/tailwind.config.js @@ -1,6 +1,6 @@ /** @type {import('tailwindcss').Config} */ module.exports = { - content: ["./src/templates/**/*.{html,js}"], + content: ["./src/templates/**/*.{html,js,ejs}", "./static/**/*.{html,js,ejs}"], theme: { extend: {}, },