diff --git a/.env.example b/.env.example index 68d322b..2721dec 100644 --- a/.env.example +++ b/.env.example @@ -6,10 +6,10 @@ JWT_SECRET="secret" CLIENT_URL="http://localhost:5173" # Database configuration -REDIS_URL="redis://@redis:6379/4" -DB_HOST="mariadb" -DB_USER="mariadb" -DB_PASS="mariadb" +REDIS_URL="redis://@127.0.0.1:6379/4" +DB_HOST="localhost" +DB_USER="root" +DB_PASS="" DB_PORT="3306" DB_NAME="game" diff --git a/package-lock.json b/package-lock.json index 8283357..403dcff 100644 --- a/package-lock.json +++ b/package-lock.json @@ -59,14 +59,14 @@ } }, "node_modules/@babel/generator": { - "version": "7.26.9", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.26.9.tgz", - "integrity": "sha512-kEWdzjOAUMW4hAyrzJ0ZaTOu9OmpyDIQicIh0zg0EEcEkYXZb2TjtBhnHi2ViX7PKwZqF4xwqfAm299/QMP3lg==", + "version": "7.26.10", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.26.10.tgz", + "integrity": "sha512-rRHT8siFIXQrAYOYqZQVsAr8vJ+cBNqcVAY6m5V8/4QqzaPl+zDBe6cLEPRDuNOUf3ww8RfJVlOyQMoSI+5Ang==", "dev": true, "license": "MIT", "dependencies": { - "@babel/parser": "^7.26.9", - "@babel/types": "^7.26.9", + "@babel/parser": "^7.26.10", + "@babel/types": "^7.26.10", "@jridgewell/gen-mapping": "^0.3.5", "@jridgewell/trace-mapping": "^0.3.25", "jsesc": "^3.0.2" @@ -96,13 +96,13 @@ } }, "node_modules/@babel/parser": { - "version": "7.26.9", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.26.9.tgz", - "integrity": "sha512-81NWa1njQblgZbQHxWHpxxCzNsa3ZwvFqpUg7P+NNUU6f3UU2jBEg4OlF/J6rl8+PQGh1q6/zWScd001YwcA5A==", + "version": "7.26.10", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.26.10.tgz", + "integrity": "sha512-6aQR2zGE/QFi8JpDLjUZEPYOs7+mhKXm86VaKFiLP35JQwQb6bwUE+XbvkH0EptsYhbNBSUGaUBLKqxH1xSgsA==", "dev": true, "license": "MIT", "dependencies": { - "@babel/types": "^7.26.9" + "@babel/types": "^7.26.10" }, "bin": { "parser": "bin/babel-parser.js" @@ -127,17 +127,17 @@ } }, "node_modules/@babel/traverse": { - "version": "7.26.9", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.26.9.tgz", - "integrity": "sha512-ZYW7L+pL8ahU5fXmNbPF+iZFHCv5scFak7MZ9bwaRPLUhHh7QQEMjZUg0HevihoqCM5iSYHN61EyCoZvqC+bxg==", + "version": "7.26.10", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.26.10.tgz", + "integrity": "sha512-k8NuDrxr0WrPH5Aupqb2LCVURP/S0vBEn5mK6iH+GIYob66U5EtoZvcdudR2jQ4cmTwhEwW1DLB+Yyas9zjF6A==", "dev": true, "license": "MIT", "dependencies": { "@babel/code-frame": "^7.26.2", - "@babel/generator": "^7.26.9", - "@babel/parser": "^7.26.9", + "@babel/generator": "^7.26.10", + "@babel/parser": "^7.26.10", "@babel/template": "^7.26.9", - "@babel/types": "^7.26.9", + "@babel/types": "^7.26.10", "debug": "^4.3.1", "globals": "^11.1.0" }, @@ -146,9 +146,9 @@ } }, "node_modules/@babel/types": { - "version": "7.26.9", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.26.9.tgz", - "integrity": "sha512-Y3IR1cRnOxOCDvMmNiym7XpXQ93iGDDPHx+Zj+NM+rg0fBaShfQLkg+hKPaZCEvg5N/LeCo4+Rj/i3FuJsIQaw==", + "version": "7.26.10", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.26.10.tgz", + "integrity": "sha512-emqcG3vHrpxUKTrxcblR36dcrcoRDvKmnL/dCL6ZsHaShW80qxCAcNhzQZrpeM765VzEos+xOi4s+r4IXzTwdQ==", "dev": true, "license": "MIT", "dependencies": { @@ -194,9 +194,9 @@ } }, "node_modules/@esbuild/aix-ppc64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.23.1.tgz", - "integrity": "sha512-6VhYk1diRqrhBAqpJEdjASR/+WVRtfjpqKuNw11cLiaWpAT/Uu+nokB+UJnevzy/P9C/ty6AOe0dwueMrGh/iQ==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.25.1.tgz", + "integrity": "sha512-kfYGy8IdzTGy+z0vFGvExZtxkFlA4zAxgKEahG9KE1ScBjpQnFsNOX8KTU5ojNru5ed5CVoJYXFtoxaq5nFbjQ==", "cpu": [ "ppc64" ], @@ -211,9 +211,9 @@ } }, "node_modules/@esbuild/android-arm": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.23.1.tgz", - "integrity": "sha512-uz6/tEy2IFm9RYOyvKl88zdzZfwEfKZmnX9Cj1BHjeSGNuGLuMD1kR8y5bteYmwqKm1tj8m4cb/aKEorr6fHWQ==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.25.1.tgz", + "integrity": "sha512-dp+MshLYux6j/JjdqVLnMglQlFu+MuVeNrmT5nk6q07wNhCdSnB7QZj+7G8VMUGh1q+vj2Bq8kRsuyA00I/k+Q==", "cpu": [ "arm" ], @@ -228,9 +228,9 @@ } }, "node_modules/@esbuild/android-arm64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.23.1.tgz", - "integrity": "sha512-xw50ipykXcLstLeWH7WRdQuysJqejuAGPd30vd1i5zSyKK3WE+ijzHmLKxdiCMtH1pHz78rOg0BKSYOSB/2Khw==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.25.1.tgz", + "integrity": "sha512-50tM0zCJW5kGqgG7fQ7IHvQOcAn9TKiVRuQ/lN0xR+T2lzEFvAi1ZcS8DiksFcEpf1t/GYOeOfCAgDHFpkiSmA==", "cpu": [ "arm64" ], @@ -245,9 +245,9 @@ } }, "node_modules/@esbuild/android-x64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.23.1.tgz", - "integrity": "sha512-nlN9B69St9BwUoB+jkyU090bru8L0NA3yFvAd7k8dNsVH8bi9a8cUAUSEcEEgTp2z3dbEDGJGfP6VUnkQnlReg==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.25.1.tgz", + "integrity": "sha512-GCj6WfUtNldqUzYkN/ITtlhwQqGWu9S45vUXs7EIYf+7rCiiqH9bCloatO9VhxsL0Pji+PF4Lz2XXCES+Q8hDw==", "cpu": [ "x64" ], @@ -262,9 +262,9 @@ } }, "node_modules/@esbuild/darwin-arm64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.23.1.tgz", - "integrity": "sha512-YsS2e3Wtgnw7Wq53XXBLcV6JhRsEq8hkfg91ESVadIrzr9wO6jJDMZnCQbHm1Guc5t/CdDiFSSfWP58FNuvT3Q==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.1.tgz", + "integrity": "sha512-5hEZKPf+nQjYoSr/elb62U19/l1mZDdqidGfmFutVUjjUZrOazAtwK+Kr+3y0C/oeJfLlxo9fXb1w7L+P7E4FQ==", "cpu": [ "arm64" ], @@ -279,9 +279,9 @@ } }, "node_modules/@esbuild/darwin-x64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.23.1.tgz", - "integrity": "sha512-aClqdgTDVPSEGgoCS8QDG37Gu8yc9lTHNAQlsztQ6ENetKEO//b8y31MMu2ZaPbn4kVsIABzVLXYLhCGekGDqw==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.25.1.tgz", + "integrity": "sha512-hxVnwL2Dqs3fM1IWq8Iezh0cX7ZGdVhbTfnOy5uURtao5OIVCEyj9xIzemDi7sRvKsuSdtCAhMKarxqtlyVyfA==", "cpu": [ "x64" ], @@ -296,9 +296,9 @@ } }, "node_modules/@esbuild/freebsd-arm64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.23.1.tgz", - "integrity": "sha512-h1k6yS8/pN/NHlMl5+v4XPfikhJulk4G+tKGFIOwURBSFzE8bixw1ebjluLOjfwtLqY0kewfjLSrO6tN2MgIhA==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.1.tgz", + "integrity": "sha512-1MrCZs0fZa2g8E+FUo2ipw6jw5qqQiH+tERoS5fAfKnRx6NXH31tXBKI3VpmLijLH6yriMZsxJtaXUyFt/8Y4A==", "cpu": [ "arm64" ], @@ -313,9 +313,9 @@ } }, "node_modules/@esbuild/freebsd-x64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.23.1.tgz", - "integrity": "sha512-lK1eJeyk1ZX8UklqFd/3A60UuZ/6UVfGT2LuGo3Wp4/z7eRTRYY+0xOu2kpClP+vMTi9wKOfXi2vjUpO1Ro76g==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.25.1.tgz", + "integrity": "sha512-0IZWLiTyz7nm0xuIs0q1Y3QWJC52R8aSXxe40VUxm6BB1RNmkODtW6LHvWRrGiICulcX7ZvyH6h5fqdLu4gkww==", "cpu": [ "x64" ], @@ -330,9 +330,9 @@ } }, "node_modules/@esbuild/linux-arm": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.23.1.tgz", - "integrity": "sha512-CXXkzgn+dXAPs3WBwE+Kvnrf4WECwBdfjfeYHpMeVxWE0EceB6vhWGShs6wi0IYEqMSIzdOF1XjQ/Mkm5d7ZdQ==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.25.1.tgz", + "integrity": "sha512-NdKOhS4u7JhDKw9G3cY6sWqFcnLITn6SqivVArbzIaf3cemShqfLGHYMx8Xlm/lBit3/5d7kXvriTUGa5YViuQ==", "cpu": [ "arm" ], @@ -347,9 +347,9 @@ } }, "node_modules/@esbuild/linux-arm64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.23.1.tgz", - "integrity": "sha512-/93bf2yxencYDnItMYV/v116zff6UyTjo4EtEQjUBeGiVpMmffDNUyD9UN2zV+V3LRV3/on4xdZ26NKzn6754g==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.25.1.tgz", + "integrity": "sha512-jaN3dHi0/DDPelk0nLcXRm1q7DNJpjXy7yWaWvbfkPvI+7XNSc/lDOnCLN7gzsyzgu6qSAmgSvP9oXAhP973uQ==", "cpu": [ "arm64" ], @@ -364,9 +364,9 @@ } }, "node_modules/@esbuild/linux-ia32": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.23.1.tgz", - "integrity": "sha512-VTN4EuOHwXEkXzX5nTvVY4s7E/Krz7COC8xkftbbKRYAl96vPiUssGkeMELQMOnLOJ8k3BY1+ZY52tttZnHcXQ==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.25.1.tgz", + "integrity": "sha512-OJykPaF4v8JidKNGz8c/q1lBO44sQNUQtq1KktJXdBLn1hPod5rE/Hko5ugKKZd+D2+o1a9MFGUEIUwO2YfgkQ==", "cpu": [ "ia32" ], @@ -381,9 +381,9 @@ } }, "node_modules/@esbuild/linux-loong64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.23.1.tgz", - "integrity": "sha512-Vx09LzEoBa5zDnieH8LSMRToj7ir/Jeq0Gu6qJ/1GcBq9GkfoEAoXvLiW1U9J1qE/Y/Oyaq33w5p2ZWrNNHNEw==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.25.1.tgz", + "integrity": "sha512-nGfornQj4dzcq5Vp835oM/o21UMlXzn79KobKlcs3Wz9smwiifknLy4xDCLUU0BWp7b/houtdrgUz7nOGnfIYg==", "cpu": [ "loong64" ], @@ -398,9 +398,9 @@ } }, "node_modules/@esbuild/linux-mips64el": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.23.1.tgz", - "integrity": "sha512-nrFzzMQ7W4WRLNUOU5dlWAqa6yVeI0P78WKGUo7lg2HShq/yx+UYkeNSE0SSfSure0SqgnsxPvmAUu/vu0E+3Q==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.25.1.tgz", + "integrity": "sha512-1osBbPEFYwIE5IVB/0g2X6i1qInZa1aIoj1TdL4AaAb55xIIgbg8Doq6a5BzYWgr+tEcDzYH67XVnTmUzL+nXg==", "cpu": [ "mips64el" ], @@ -415,9 +415,9 @@ } }, "node_modules/@esbuild/linux-ppc64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.23.1.tgz", - "integrity": "sha512-dKN8fgVqd0vUIjxuJI6P/9SSSe/mB9rvA98CSH2sJnlZ/OCZWO1DJvxj8jvKTfYUdGfcq2dDxoKaC6bHuTlgcw==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.25.1.tgz", + "integrity": "sha512-/6VBJOwUf3TdTvJZ82qF3tbLuWsscd7/1w+D9LH0W/SqUgM5/JJD0lrJ1fVIfZsqB6RFmLCe0Xz3fmZc3WtyVg==", "cpu": [ "ppc64" ], @@ -432,9 +432,9 @@ } }, "node_modules/@esbuild/linux-riscv64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.23.1.tgz", - "integrity": "sha512-5AV4Pzp80fhHL83JM6LoA6pTQVWgB1HovMBsLQ9OZWLDqVY8MVobBXNSmAJi//Csh6tcY7e7Lny2Hg1tElMjIA==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.25.1.tgz", + "integrity": "sha512-nSut/Mx5gnilhcq2yIMLMe3Wl4FK5wx/o0QuuCLMtmJn+WeWYoEGDN1ipcN72g1WHsnIbxGXd4i/MF0gTcuAjQ==", "cpu": [ "riscv64" ], @@ -449,9 +449,9 @@ } }, "node_modules/@esbuild/linux-s390x": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.23.1.tgz", - "integrity": "sha512-9ygs73tuFCe6f6m/Tb+9LtYxWR4c9yg7zjt2cYkjDbDpV/xVn+68cQxMXCjUpYwEkze2RcU/rMnfIXNRFmSoDw==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.25.1.tgz", + "integrity": "sha512-cEECeLlJNfT8kZHqLarDBQso9a27o2Zd2AQ8USAEoGtejOrCYHNtKP8XQhMDJMtthdF4GBmjR2au3x1udADQQQ==", "cpu": [ "s390x" ], @@ -466,9 +466,9 @@ } }, "node_modules/@esbuild/linux-x64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.23.1.tgz", - "integrity": "sha512-EV6+ovTsEXCPAp58g2dD68LxoP/wK5pRvgy0J/HxPGB009omFPv3Yet0HiaqvrIrgPTBuC6wCH1LTOY91EO5hQ==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.1.tgz", + "integrity": "sha512-xbfUhu/gnvSEg+EGovRc+kjBAkrvtk38RlerAzQxvMzlB4fXpCFCeUAYzJvrnhFtdeyVCDANSjJvOvGYoeKzFA==", "cpu": [ "x64" ], @@ -482,10 +482,27 @@ "node": ">=18" } }, + "node_modules/@esbuild/netbsd-arm64": { + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.1.tgz", + "integrity": "sha512-O96poM2XGhLtpTh+s4+nP7YCCAfb4tJNRVZHfIE7dgmax+yMP2WgMd2OecBuaATHKTHsLWHQeuaxMRnCsH8+5g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, "node_modules/@esbuild/netbsd-x64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.23.1.tgz", - "integrity": "sha512-aevEkCNu7KlPRpYLjwmdcuNz6bDFiE7Z8XC4CPqExjTvrHugh28QzUXVOZtiYghciKUacNktqxdpymplil1beA==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.25.1.tgz", + "integrity": "sha512-X53z6uXip6KFXBQ+Krbx25XHV/NCbzryM6ehOAeAil7X7oa4XIq+394PWGnwaSQ2WRA0KI6PUO6hTO5zeF5ijA==", "cpu": [ "x64" ], @@ -500,9 +517,9 @@ } }, "node_modules/@esbuild/openbsd-arm64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.23.1.tgz", - "integrity": "sha512-3x37szhLexNA4bXhLrCC/LImN/YtWis6WXr1VESlfVtVeoFJBRINPJ3f0a/6LV8zpikqoUg4hyXw0sFBt5Cr+Q==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.1.tgz", + "integrity": "sha512-Na9T3szbXezdzM/Kfs3GcRQNjHzM6GzFBeU1/6IV/npKP5ORtp9zbQjvkDJ47s6BCgaAZnnnu/cY1x342+MvZg==", "cpu": [ "arm64" ], @@ -517,9 +534,9 @@ } }, "node_modules/@esbuild/openbsd-x64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.23.1.tgz", - "integrity": "sha512-aY2gMmKmPhxfU+0EdnN+XNtGbjfQgwZj43k8G3fyrDM/UdZww6xrWxmDkuz2eCZchqVeABjV5BpildOrUbBTqA==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.25.1.tgz", + "integrity": "sha512-T3H78X2h1tszfRSf+txbt5aOp/e7TAz3ptVKu9Oyir3IAOFPGV6O9c2naym5TOriy1l0nNf6a4X5UXRZSGX/dw==", "cpu": [ "x64" ], @@ -534,9 +551,9 @@ } }, "node_modules/@esbuild/sunos-x64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.23.1.tgz", - "integrity": "sha512-RBRT2gqEl0IKQABT4XTj78tpk9v7ehp+mazn2HbUeZl1YMdaGAQqhapjGTCe7uw7y0frDi4gS0uHzhvpFuI1sA==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.25.1.tgz", + "integrity": "sha512-2H3RUvcmULO7dIE5EWJH8eubZAI4xw54H1ilJnRNZdeo8dTADEZ21w6J22XBkXqGJbe0+wnNJtw3UXRoLJnFEg==", "cpu": [ "x64" ], @@ -551,9 +568,9 @@ } }, "node_modules/@esbuild/win32-arm64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.23.1.tgz", - "integrity": "sha512-4O+gPR5rEBe2FpKOVyiJ7wNDPA8nGzDuJ6gN4okSA1gEOYZ67N8JPk58tkWtdtPeLz7lBnY6I5L3jdsr3S+A6A==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.25.1.tgz", + "integrity": "sha512-GE7XvrdOzrb+yVKB9KsRMq+7a2U/K5Cf/8grVFRAGJmfADr/e/ODQ134RK2/eeHqYV5eQRFxb1hY7Nr15fv1NQ==", "cpu": [ "arm64" ], @@ -568,9 +585,9 @@ } }, "node_modules/@esbuild/win32-ia32": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.23.1.tgz", - "integrity": "sha512-BcaL0Vn6QwCwre3Y717nVHZbAa4UBEigzFm6VdsVdT/MbZ38xoj1X9HPkZhbmaBGUD1W8vxAfffbDe8bA6AKnQ==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.25.1.tgz", + "integrity": "sha512-uOxSJCIcavSiT6UnBhBzE8wy3n0hOkJsBOzy7HDAuTDE++1DJMRRVCPGisULScHL+a/ZwdXPpXD3IyFKjA7K8A==", "cpu": [ "ia32" ], @@ -585,9 +602,9 @@ } }, "node_modules/@esbuild/win32-x64": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.23.1.tgz", - "integrity": "sha512-BHpFFeslkWrXWyUPnbKm+xYYVYruCinGcftSBaa8zoF9hZO4BcSCFUvHVTtzpIY6YzUnYtuEhZ+C9iEXjxnasg==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.25.1.tgz", + "integrity": "sha512-Y1EQdcfwMSeQN/ujR5VayLOJ1BHaK+ssyk0AEzPjC+t1lITgsnccPqFjb6V+LsTp/9Iov4ysfjxLaGJ9RPtkVg==", "cpu": [ "x64" ], @@ -1056,14 +1073,14 @@ } }, "node_modules/@mikro-orm/cli": { - "version": "6.4.6", - "resolved": "https://registry.npmjs.org/@mikro-orm/cli/-/cli-6.4.6.tgz", - "integrity": "sha512-sTMoDSJrnHZBT+ZAG40OeZwR9zRTYHtaaub9OoMM2CrxfI1KeiNqL/XFB4LaM5SVRAbnoEFpMJwQ8KS+5NcN9w==", + "version": "6.4.9", + "resolved": "https://registry.npmjs.org/@mikro-orm/cli/-/cli-6.4.9.tgz", + "integrity": "sha512-LQzVsmar/0DoJkPGyz3OpB8pa9BCQtvYreEC71h0O+RcizppJjgBQNTkj5tJd2Iqvh4hSaMv6qTv0l5UK6F2Vw==", "license": "MIT", "dependencies": { "@jercle/yargonaut": "1.1.5", - "@mikro-orm/core": "6.4.6", - "@mikro-orm/knex": "6.4.6", + "@mikro-orm/core": "6.4.9", + "@mikro-orm/knex": "6.4.9", "fs-extra": "11.3.0", "tsconfig-paths": "4.2.0", "yargs": "17.7.2" @@ -1077,9 +1094,9 @@ } }, "node_modules/@mikro-orm/core": { - "version": "6.4.6", - "resolved": "https://registry.npmjs.org/@mikro-orm/core/-/core-6.4.6.tgz", - "integrity": "sha512-xVm/ALG/3vTMgh6SrvojJ6jjMa0s2hNzWN0triDB16BaNdLwWE4aAaAe+3CuoMFqJAArSOUISTEjExbzELB1ZA==", + "version": "6.4.9", + "resolved": "https://registry.npmjs.org/@mikro-orm/core/-/core-6.4.9.tgz", + "integrity": "sha512-osB2TbvSH4ZL1s62LCBQFAnxPqLycX5fakPHOoztudixqfbVD5QQydeGizJXMMh2zKP6vRCwIJy3MeSuFxPjHg==", "license": "MIT", "dependencies": { "dataloader": "2.2.3", @@ -1087,7 +1104,7 @@ "esprima": "4.0.1", "fs-extra": "11.3.0", "globby": "11.1.0", - "mikro-orm": "6.4.6", + "mikro-orm": "6.4.9", "reflect-metadata": "0.2.2" }, "engines": { @@ -1098,9 +1115,9 @@ } }, "node_modules/@mikro-orm/knex": { - "version": "6.4.6", - "resolved": "https://registry.npmjs.org/@mikro-orm/knex/-/knex-6.4.6.tgz", - "integrity": "sha512-o6t67tFH/GuPZCCEtKbTTL8HDXNgB2ITjButCTZLwteL0qI9yE/f7K6K+dEUKW+hAL3KRvc2BQeumvCVWFeISg==", + "version": "6.4.9", + "resolved": "https://registry.npmjs.org/@mikro-orm/knex/-/knex-6.4.9.tgz", + "integrity": "sha512-iGXJfe/TziVOQsWuxMIqkOpurysWzQA6kj3+FDtOkHJAijZhqhjSBnfUVHHY/JzU9o0M0rgLrDVJFry/uEaJEA==", "license": "MIT", "dependencies": { "fs-extra": "11.3.0", @@ -1129,12 +1146,12 @@ } }, "node_modules/@mikro-orm/mariadb": { - "version": "6.4.6", - "resolved": "https://registry.npmjs.org/@mikro-orm/mariadb/-/mariadb-6.4.6.tgz", - "integrity": "sha512-n6pOf69heOsbrggqYcf9SeF9hUdkw0FbzuUAcI72jWuyNRyzNR1UATblD+vRJnwt8JDWwakjINU/bduZbcEwPw==", + "version": "6.4.9", + "resolved": "https://registry.npmjs.org/@mikro-orm/mariadb/-/mariadb-6.4.9.tgz", + "integrity": "sha512-KuCzDGkC9cmNA8WxE9pZca6/Ds2sso3JUxiGoVyekOj/t9qer81UQYWasI80TBJ82TxrUdLM9NFBBO++tz+NYA==", "license": "MIT", "dependencies": { - "@mikro-orm/knex": "6.4.6", + "@mikro-orm/knex": "6.4.9", "mariadb": "3.4.0" }, "engines": { @@ -1145,12 +1162,12 @@ } }, "node_modules/@mikro-orm/migrations": { - "version": "6.4.6", - "resolved": "https://registry.npmjs.org/@mikro-orm/migrations/-/migrations-6.4.6.tgz", - "integrity": "sha512-i0/H07g1jQS0tKVSTSkHhrmuDEHxDD3/IzkiObezTgGlD5tqN7acaSr8RDJ3DgICb8MHUDVMLwxeGy8igDB4ag==", + "version": "6.4.9", + "resolved": "https://registry.npmjs.org/@mikro-orm/migrations/-/migrations-6.4.9.tgz", + "integrity": "sha512-vwTXG8PU3bpzTZxxu1dWlhnUumM2Yob2IWajoh+Rj9+19VBZYc5N3tm8FRekt5oPzxeK5/9+sDfxT9FzXgjeNw==", "license": "MIT", "dependencies": { - "@mikro-orm/knex": "6.4.6", + "@mikro-orm/knex": "6.4.9", "fs-extra": "11.3.0", "umzug": "3.8.2" }, @@ -1162,13 +1179,13 @@ } }, "node_modules/@mikro-orm/mysql": { - "version": "6.4.6", - "resolved": "https://registry.npmjs.org/@mikro-orm/mysql/-/mysql-6.4.6.tgz", - "integrity": "sha512-KVP9Wif9MX/RrroVgYQQUrXe9SALBQLfB9CbuJlUB7MnEcZtDi5JNX7z5kghToz0aBrTtOgsr93G1bCoM0SJkg==", + "version": "6.4.9", + "resolved": "https://registry.npmjs.org/@mikro-orm/mysql/-/mysql-6.4.9.tgz", + "integrity": "sha512-rmHonMzvurB+50BNpKb9FORFVs3+V8S4Om1Tmv6MFvSdeD1Qqb/efvYp7cgv+NncHSrgMtKLMy3FDm7guU6yYQ==", "license": "MIT", "dependencies": { - "@mikro-orm/knex": "6.4.6", - "mysql2": "3.12.0" + "@mikro-orm/knex": "6.4.9", + "mysql2": "3.13.0" }, "engines": { "node": ">= 18.12.0" @@ -1178,9 +1195,9 @@ } }, "node_modules/@mikro-orm/reflection": { - "version": "6.4.6", - "resolved": "https://registry.npmjs.org/@mikro-orm/reflection/-/reflection-6.4.6.tgz", - "integrity": "sha512-7mL7HFVnaOOhDNgLjjndWyeJUtOl2wKn0spSqB8uRjS4XtwNEGVZNkW5YD1t/x7TJ99wUhe+oRDiySciiJSeBQ==", + "version": "6.4.9", + "resolved": "https://registry.npmjs.org/@mikro-orm/reflection/-/reflection-6.4.9.tgz", + "integrity": "sha512-fgY7yLrcZm3J/8dv9reUC4PQo7C2muImU31jmzz1SxmNKPJFDJl7OzcDZlM5NOisXzsWUBrcNdCyuQiWViVc3A==", "license": "MIT", "dependencies": { "globby": "11.1.0", @@ -1307,9 +1324,9 @@ } }, "node_modules/@rushstack/node-core-library": { - "version": "5.11.0", - "resolved": "https://registry.npmjs.org/@rushstack/node-core-library/-/node-core-library-5.11.0.tgz", - "integrity": "sha512-I8+VzG9A0F3nH2rLpPd7hF8F7l5Xb7D+ldrWVZYegXM6CsKkvWc670RlgK3WX8/AseZfXA/vVrh0bpXe2Y2UDQ==", + "version": "5.12.0", + "resolved": "https://registry.npmjs.org/@rushstack/node-core-library/-/node-core-library-5.12.0.tgz", + "integrity": "sha512-QSwwzgzWoil1SCQse+yCHwlhRxNv2dX9siPnAb9zR/UmMhac4mjMrlMZpk64BlCeOFi1kJKgXRkihSwRMbboAQ==", "license": "MIT", "dependencies": { "ajv": "~8.13.0", @@ -1358,12 +1375,12 @@ } }, "node_modules/@rushstack/terminal": { - "version": "0.15.0", - "resolved": "https://registry.npmjs.org/@rushstack/terminal/-/terminal-0.15.0.tgz", - "integrity": "sha512-vXQPRQ+vJJn4GVqxkwRe+UGgzNxdV8xuJZY2zem46Y0p3tlahucH9/hPmLGj2i9dQnUBFiRnoM9/KW7PYw8F4Q==", + "version": "0.15.1", + "resolved": "https://registry.npmjs.org/@rushstack/terminal/-/terminal-0.15.1.tgz", + "integrity": "sha512-3vgJYwumcjoDOXU3IxZfd616lqOdmr8Ezj4OWgJZfhmiBK4Nh7eWcv8sU8N/HdzXcuHDXCRGn/6O2Q75QvaZMA==", "license": "MIT", "dependencies": { - "@rushstack/node-core-library": "5.11.0", + "@rushstack/node-core-library": "5.12.0", "supports-color": "~8.1.1" }, "peerDependencies": { @@ -1391,12 +1408,12 @@ } }, "node_modules/@rushstack/ts-command-line": { - "version": "4.23.5", - "resolved": "https://registry.npmjs.org/@rushstack/ts-command-line/-/ts-command-line-4.23.5.tgz", - "integrity": "sha512-jg70HfoK44KfSP3MTiL5rxsZH7X1ktX3cZs9Sl8eDu1/LxJSbPsh0MOFRC710lIuYYSgxWjI5AjbCBAl7u3RxA==", + "version": "4.23.6", + "resolved": "https://registry.npmjs.org/@rushstack/ts-command-line/-/ts-command-line-4.23.6.tgz", + "integrity": "sha512-7WepygaF3YPEoToh4MAL/mmHkiIImQq3/uAkQX46kVoKTNOOlCtFGyNnze6OYuWw2o9rxsyrHVfIBKxq/am2RA==", "license": "MIT", "dependencies": { - "@rushstack/terminal": "0.15.0", + "@rushstack/terminal": "0.15.1", "@types/argparse": "1.0.38", "argparse": "~1.0.9", "string-argv": "~0.3.1" @@ -1563,9 +1580,9 @@ } }, "node_modules/@types/jsonwebtoken": { - "version": "9.0.8", - "resolved": "https://registry.npmjs.org/@types/jsonwebtoken/-/jsonwebtoken-9.0.8.tgz", - "integrity": "sha512-7fx54m60nLFUVYlxAB1xpe9CBWX2vSrk50Y6ogRJ1v5xxtba7qXTg5BgYDN5dq+yuQQ9HaVlHJyAAt1/mxryFg==", + "version": "9.0.9", + "resolved": "https://registry.npmjs.org/@types/jsonwebtoken/-/jsonwebtoken-9.0.9.tgz", + "integrity": "sha512-uoe+GxEuHbvy12OUQct2X9JenKM3qAscquYymuQN4fMWG9DBQtykrQEFcAbVACF7qaLw9BePSodUL0kquqBJpQ==", "dev": true, "license": "MIT", "dependencies": { @@ -1588,9 +1605,9 @@ "license": "MIT" }, "node_modules/@types/node": { - "version": "20.17.19", - "resolved": "https://registry.npmjs.org/@types/node/-/node-20.17.19.tgz", - "integrity": "sha512-LEwC7o1ifqg/6r2gn9Dns0f1rhK+fPFDoMiceTJ6kWmVk6bgXBI/9IOWfVan4WiAavK9pIVWdX0/e3J+eEUh5A==", + "version": "20.17.24", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.17.24.tgz", + "integrity": "sha512-d7fGCyB96w9BnWQrOsJtpyiSaBcAYYr75bnK6ZRjDbql2cGLj/3GsL5OYmLPNq76l7Gf2q4Rv9J2o6h5CrD9sA==", "license": "MIT", "dependencies": { "undici-types": "~6.19.2" @@ -1657,9 +1674,9 @@ } }, "node_modules/acorn": { - "version": "8.14.0", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.14.0.tgz", - "integrity": "sha512-cl669nCJTZBsL97OF4kUQm5g5hC2uihk0NxY3WENAC0TYdILVkAyHymAntgxGkl7K+t0cXIrH5siy5S4XkFycA==", + "version": "8.14.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.14.1.tgz", + "integrity": "sha512-OvQ/2pUDKmgfCg++xsTX1wGxfTaszcHVcTctW4UJB4hibJx2HXxxO5UmVgyjMa+ZDsiaf5wWLXYpRWMmBI0QHg==", "dev": true, "license": "MIT", "bin": { @@ -1932,9 +1949,9 @@ } }, "node_modules/bullmq": { - "version": "5.41.0", - "resolved": "https://registry.npmjs.org/bullmq/-/bullmq-5.41.0.tgz", - "integrity": "sha512-GGfKu2DHGIvbnMtQjR/82wvWsdCaGxN5JGR3pvKd1mkDI9DsWn8r0+pAzZ6Y4ImWXFaetaAqywOhv2Ik0R2m3g==", + "version": "5.41.9", + "resolved": "https://registry.npmjs.org/bullmq/-/bullmq-5.41.9.tgz", + "integrity": "sha512-dcmBEo6CzLh3PQ7KN6qz5M8VsSv49vcXfEadgrUdfmUvYDpg630/yPV/Ov18unPT4IOCjz1XZSgpjs5DjwlBYw==", "license": "MIT", "dependencies": { "cron-parser": "^4.9.0", @@ -1969,13 +1986,13 @@ } }, "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==", + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz", + "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==", "license": "MIT", "dependencies": { - "call-bind-apply-helpers": "^1.0.1", - "get-intrinsic": "^1.2.6" + "call-bind-apply-helpers": "^1.0.2", + "get-intrinsic": "^1.3.0" }, "engines": { "node": ">= 0.4" @@ -2421,9 +2438,9 @@ } }, "node_modules/esbuild": { - "version": "0.23.1", - "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.23.1.tgz", - "integrity": "sha512-VVNz/9Sa0bs5SELtn3f7qhJCDPCF5oMEl5cO9/SSinpE9hbPVvxbd572HH5AKiP7WD8INO53GgfDDhRjkylHEg==", + "version": "0.25.1", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.1.tgz", + "integrity": "sha512-BGO5LtrGC7vxnqucAe/rmvKdJllfGaYWdyABvyMoXQlfYMb2bbRuReWR5tEGE//4LcNJj9XrkovTqNYRFZHAMQ==", "dev": true, "hasInstallScript": true, "license": "MIT", @@ -2434,30 +2451,31 @@ "node": ">=18" }, "optionalDependencies": { - "@esbuild/aix-ppc64": "0.23.1", - "@esbuild/android-arm": "0.23.1", - "@esbuild/android-arm64": "0.23.1", - "@esbuild/android-x64": "0.23.1", - "@esbuild/darwin-arm64": "0.23.1", - "@esbuild/darwin-x64": "0.23.1", - "@esbuild/freebsd-arm64": "0.23.1", - "@esbuild/freebsd-x64": "0.23.1", - "@esbuild/linux-arm": "0.23.1", - "@esbuild/linux-arm64": "0.23.1", - "@esbuild/linux-ia32": "0.23.1", - "@esbuild/linux-loong64": "0.23.1", - "@esbuild/linux-mips64el": "0.23.1", - "@esbuild/linux-ppc64": "0.23.1", - "@esbuild/linux-riscv64": "0.23.1", - "@esbuild/linux-s390x": "0.23.1", - "@esbuild/linux-x64": "0.23.1", - "@esbuild/netbsd-x64": "0.23.1", - "@esbuild/openbsd-arm64": "0.23.1", - "@esbuild/openbsd-x64": "0.23.1", - "@esbuild/sunos-x64": "0.23.1", - "@esbuild/win32-arm64": "0.23.1", - "@esbuild/win32-ia32": "0.23.1", - "@esbuild/win32-x64": "0.23.1" + "@esbuild/aix-ppc64": "0.25.1", + "@esbuild/android-arm": "0.25.1", + "@esbuild/android-arm64": "0.25.1", + "@esbuild/android-x64": "0.25.1", + "@esbuild/darwin-arm64": "0.25.1", + "@esbuild/darwin-x64": "0.25.1", + "@esbuild/freebsd-arm64": "0.25.1", + "@esbuild/freebsd-x64": "0.25.1", + "@esbuild/linux-arm": "0.25.1", + "@esbuild/linux-arm64": "0.25.1", + "@esbuild/linux-ia32": "0.25.1", + "@esbuild/linux-loong64": "0.25.1", + "@esbuild/linux-mips64el": "0.25.1", + "@esbuild/linux-ppc64": "0.25.1", + "@esbuild/linux-riscv64": "0.25.1", + "@esbuild/linux-s390x": "0.25.1", + "@esbuild/linux-x64": "0.25.1", + "@esbuild/netbsd-arm64": "0.25.1", + "@esbuild/netbsd-x64": "0.25.1", + "@esbuild/openbsd-arm64": "0.25.1", + "@esbuild/openbsd-x64": "0.25.1", + "@esbuild/sunos-x64": "0.25.1", + "@esbuild/win32-arm64": "0.25.1", + "@esbuild/win32-ia32": "0.25.1", + "@esbuild/win32-x64": "0.25.1" } }, "node_modules/escalade": { @@ -2599,9 +2617,9 @@ } }, "node_modules/fastq": { - "version": "1.19.0", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.0.tgz", - "integrity": "sha512-7SFSRCNjBQIZH/xZR3iy5iQYR8aGBE0h3VG6/cwlbrpdciNYBMotQav8c1XI3HjHH+NikUpP53nPdlZSdWmFzA==", + "version": "1.19.1", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.1.tgz", + "integrity": "sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ==", "license": "ISC", "dependencies": { "reusify": "^1.0.4" @@ -2739,17 +2757,17 @@ } }, "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==", + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz", + "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==", "license": "MIT", "dependencies": { - "call-bind-apply-helpers": "^1.0.1", + "call-bind-apply-helpers": "^1.0.2", "es-define-property": "^1.0.1", "es-errors": "^1.3.0", - "es-object-atoms": "^1.0.0", + "es-object-atoms": "^1.1.1", "function-bind": "^1.1.2", - "get-proto": "^1.0.0", + "get-proto": "^1.0.1", "gopd": "^1.2.0", "has-symbols": "^1.1.0", "hasown": "^2.0.2", @@ -2971,9 +2989,9 @@ } }, "node_modules/ioredis": { - "version": "5.5.0", - "resolved": "https://registry.npmjs.org/ioredis/-/ioredis-5.5.0.tgz", - "integrity": "sha512-7CutT89g23FfSa8MDoIFs2GYYa0PaNiW/OrT+nRyjRXHDZd17HmIgy+reOQ/yhh72NznNjGuS8kbCAcA4Ro4mw==", + "version": "5.6.0", + "resolved": "https://registry.npmjs.org/ioredis/-/ioredis-5.6.0.tgz", + "integrity": "sha512-tBZlIIWbndeWBWCXWZiqtOF/yxf6yZX3tAlTJ7nfo5jhd6dctNxF7QnYlZLZ1a0o0pDoen7CgZqO+zjNaFbJAg==", "license": "MIT", "dependencies": { "@ioredis/commands": "^1.1.1", @@ -3316,9 +3334,9 @@ "license": "MIT" }, "node_modules/long": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/long/-/long-5.3.0.tgz", - "integrity": "sha512-5vvY5yF1zF/kXk+L94FRiTDa1Znom46UjPCH6/XbSvS8zBKMFBHTJk8KDMqJ+2J6QezQFi7k1k8v21ClJYHPaw==", + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/long/-/long-5.3.1.tgz", + "integrity": "sha512-ka87Jz3gcx/I7Hal94xaN2tZEOPoUOEVftkQqZx2EeQRN7LGdfLlI3FvZ+7WDplm+vK2Urx9ULrvSowtdCieng==", "license": "Apache-2.0" }, "node_modules/lru-cache": { @@ -3328,9 +3346,9 @@ "license": "ISC" }, "node_modules/lru.min": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/lru.min/-/lru.min-1.1.1.tgz", - "integrity": "sha512-FbAj6lXil6t8z4z3j0E5mfRlPzxkySotzUHwRXjlpRh10vc6AI6WN62ehZj82VG7M20rqogJ0GLwar2Xa05a8Q==", + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/lru.min/-/lru.min-1.1.2.tgz", + "integrity": "sha512-Nv9KddBcQSlQopmBHXSsZVY5xsdlZkdH/Iey0BlcBYggMd4two7cZnKOK9vmy3nY0O5RGH99z1PCeTpPqszUYg==", "license": "MIT", "engines": { "bun": ">=1.0.0", @@ -3375,9 +3393,9 @@ } }, "node_modules/mariadb/node_modules/@types/node": { - "version": "22.13.4", - "resolved": "https://registry.npmjs.org/@types/node/-/node-22.13.4.tgz", - "integrity": "sha512-ywP2X0DYtX3y08eFVx5fNIw7/uIv8hYUKgXoK8oayJlLnKcRfEYCxWMVE1XagUdVtCJlZT1AU4LXEABW+L1Peg==", + "version": "22.13.10", + "resolved": "https://registry.npmjs.org/@types/node/-/node-22.13.10.tgz", + "integrity": "sha512-I6LPUvlRH+O6VRUqYOcMudhaIdUVWfsjnZavnsraHvpBwaEyMN29ry+0UVJhImYL16xsscu0aske3yA+uPOWfw==", "license": "MIT", "dependencies": { "undici-types": "~6.20.0" @@ -3460,9 +3478,9 @@ } }, "node_modules/mikro-orm": { - "version": "6.4.6", - "resolved": "https://registry.npmjs.org/mikro-orm/-/mikro-orm-6.4.6.tgz", - "integrity": "sha512-Lr3uFK06O/4F/AtQAsuYD6QH7DgmUooSVFVGf1y02IuiKVFKOMJ4iKimkRMyoA+ykKhgYIp8WiaEqbWJVuz4Vw==", + "version": "6.4.9", + "resolved": "https://registry.npmjs.org/mikro-orm/-/mikro-orm-6.4.9.tgz", + "integrity": "sha512-XwVrWNT4NNwS6kHIKFNDfvy8L1eWcBBEHeTVzFFYcnb2ummATaLxqeVkNEmKA68jmdtfQdUmWBqGdbcIPwtL2Q==", "license": "MIT", "engines": { "node": ">= 18.12.0" @@ -3561,9 +3579,9 @@ } }, "node_modules/mysql2": { - "version": "3.12.0", - "resolved": "https://registry.npmjs.org/mysql2/-/mysql2-3.12.0.tgz", - "integrity": "sha512-C8fWhVysZoH63tJbX8d10IAoYCyXy4fdRFz2Ihrt9jtPILYynFEKUUzpp1U7qxzDc3tMbotvaBH+sl6bFnGZiw==", + "version": "3.13.0", + "resolved": "https://registry.npmjs.org/mysql2/-/mysql2-3.13.0.tgz", + "integrity": "sha512-M6DIQjTqKeqXH5HLbLMxwcK5XfXHw30u5ap6EZmu7QVmcF/gnh2wS/EOiQ4MTbXz/vQeoXrmycPlVRM00WSslg==", "license": "MIT", "dependencies": { "aws-ssl-profiles": "^1.1.1", @@ -3883,9 +3901,9 @@ } }, "node_modules/prettier": { - "version": "3.5.1", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.5.1.tgz", - "integrity": "sha512-hPpFQvHwL3Qv5AdRvBFMhnKo4tYxp0ReXiPn2bxkiohEX6mBeBwEpBSQTkD458RaaDKQMYSp4hX4UtfUTA5wDw==", + "version": "3.5.3", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.5.3.tgz", + "integrity": "sha512-QQtaxnoDJeAkDvDKWCLiwIXkTgRhwYDEQCghU9Z6q03iyek/rxRh/2lC3HB7P8sWT2xC/y5JDctPLBIGzHKbhw==", "dev": true, "license": "MIT", "bin": { @@ -4127,9 +4145,9 @@ } }, "node_modules/reusify": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", - "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz", + "integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==", "license": "MIT", "engines": { "iojs": ">=1.0.0", @@ -4763,13 +4781,13 @@ "license": "0BSD" }, "node_modules/tsx": { - "version": "4.19.2", - "resolved": "https://registry.npmjs.org/tsx/-/tsx-4.19.2.tgz", - "integrity": "sha512-pOUl6Vo2LUq/bSa8S5q7b91cgNSjctn9ugq/+Mvow99qW6x/UZYwzxy/3NmqoT66eHYfCVvFvACC58UBPFf28g==", + "version": "4.19.3", + "resolved": "https://registry.npmjs.org/tsx/-/tsx-4.19.3.tgz", + "integrity": "sha512-4H8vUNGNjQ4V2EOoGw005+c+dGuPSnhpPBPHBtsZdGZBk/iJb4kguGlPWaZTZ3q5nMtFOEsY0nRDlh9PJyd6SQ==", "dev": true, "license": "MIT", "dependencies": { - "esbuild": "~0.23.0", + "esbuild": "~0.25.0", "get-tsconfig": "^4.7.5" }, "bin": { @@ -4783,9 +4801,9 @@ } }, "node_modules/type-fest": { - "version": "4.34.1", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-4.34.1.tgz", - "integrity": "sha512-6kSc32kT0rbwxD6QL1CYe8IqdzN/J/ILMrNK+HMQCKH3insCDRY/3ITb0vcBss0a3t72fzh2YSzj8ko1HgwT3g==", + "version": "4.37.0", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-4.37.0.tgz", + "integrity": "sha512-S/5/0kFftkq27FPNye0XM1e2NsnoD/3FS+pBmbjmmtLT6I+i344KoOf7pvXreaFsDamWeaJX55nczA1m5PsBDg==", "license": "(MIT OR CC0-1.0)", "engines": { "node": ">=16" @@ -4808,9 +4826,9 @@ } }, "node_modules/typescript": { - "version": "5.7.3", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.7.3.tgz", - "integrity": "sha512-84MVSjMEHP+FQRPy3pX9sTVV/INIex71s9TL2Gm5FG/WG1SqXeKyZ0k7/blY/4FdOzI12CBy1vGc4og/eus0fw==", + "version": "5.8.2", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.8.2.tgz", + "integrity": "sha512-aJn6wq13/afZp/jT9QZmwEjDqqvSGp1VT5GVg+f/t6/oVyrgXM6BY1h9BRh/O5p3PlUPAe+WuiEZOmb/49RqoQ==", "license": "Apache-2.0", "bin": { "tsc": "bin/tsc", diff --git a/public/assets.zip b/public/assets.zip index a9981cf..bfbd777 100644 Binary files a/public/assets.zip and b/public/assets.zip differ diff --git a/src/application/base/baseEvent.ts b/src/application/base/baseEvent.ts index 986b149..e309089 100644 --- a/src/application/base/baseEvent.ts +++ b/src/application/base/baseEvent.ts @@ -45,15 +45,16 @@ export abstract class BaseEvent { return character?.getRole() === 'gm' } - protected emitError(message: string): void { + protected sendNotificationAndLog(message: string): void { + console.log(message) this.socket.emit(SocketEvent.NOTIFICATION, { title: 'Server message', message }) - this.logger.error('Base event error', `Player ${this.socket.userId}: ${message}`) + this.logger.error('Base event error' + `Player ${this.socket.userId}: ${message}`) } protected handleError(context: string, error: unknown): void { console.log(error) const errorMessage = error instanceof Error ? error.message : error && typeof error === 'object' && 'toString' in error ? error.toString() : String(error) this.socket.emit(SocketEvent.NOTIFICATION, { title: 'Server message', message: `Server error occured. Please contact the server administrator.` }) - this.logger.error('Base event error', errorMessage) + this.logger.error('Base event error: ' + errorMessage) } } diff --git a/src/application/enums.ts b/src/application/enums.ts index 563b4f9..9689bf9 100644 --- a/src/application/enums.ts +++ b/src/application/enums.ts @@ -1,4 +1,6 @@ export enum SocketEvent { + CONNECT_ERROR = 'connect_error', + RECONNECT_FAILED = 'reconnect_failed', CLOSE = '52', DATA = '51', CHARACTER_CONNECT = '50', @@ -35,7 +37,7 @@ export enum SocketEvent { GM_MAP_REQUEST = '19', GM_MAP_UPDATE = '18', MAP_CHARACTER_MOVEERROR = '17', - DISCONNECT = '16', + DISCONNECT = 'disconnect', USER_DISCONNECT = '15', LOGIN = '14', LOGGED_IN = '13', @@ -43,7 +45,7 @@ export enum SocketEvent { DATE = '11', FAILED = '10', COMPLETED = '9', - CONNECTION = '8', + CONNECTION = 'connection', WEATHER = '7', CHARACTER_DISCONNECT = '6', MAP_CHARACTER_ATTACK = '5', diff --git a/src/application/zodTypes.ts b/src/application/zodTypes.ts index 7252173..c4da6a9 100644 --- a/src/application/zodTypes.ts +++ b/src/application/zodTypes.ts @@ -58,3 +58,16 @@ export const ZCharacterCreate = z.object({ .max(255, { message: 'Name must be at most 255 characters long' }) .regex(/^[A-Za-z][A-Za-z0-9_-]*$/, { message: 'Name must start with a letter and can only contain letters, numbers, underscores, or dashes' }) }) + +export const ZCharacterConnect = z.object({ + characterId: z.string(), + characterHairId: z.string().optional().nullable(), + newNickname: z + .string() + .min(3, { message: 'Name must be at least 3 characters long' }) + .max(255, { message: 'Name must be at most 255 characters long' }) + .regex(/^[A-Za-z][A-Za-z0-9_-]*$/, { + message: 'Name must start with a letter and can only contain letters, numbers, underscores, or dashes' + }) + .optional() +}) diff --git a/src/commands/init.ts b/src/commands/init.ts index f20212a..d34b853 100644 --- a/src/commands/init.ts +++ b/src/commands/init.ts @@ -30,7 +30,8 @@ export default class InitCommand extends BaseCommand { // Assets await this.importTiles() await this.importMapObjects() - await this.createCharacterType() + await this.createMaleCharacterType() + // await this.createFemaleCharacterType() await this.createCharacterHair() // await this.createCharacterEquipment() @@ -74,9 +75,9 @@ export default class InitCommand extends BaseCommand { } } - private async createCharacterType(): Promise { + private async createMaleCharacterType(): Promise { const characterSprite = new Sprite() - characterSprite.setId('023d1e9d-f57f-4faa-8412-86c07107cf85').setName('Character') + characterSprite.setId('023d1e9d-f57f-4faa-8412-86c07107cf85').setName('Male character') await characterSprite.save() const idleRightDownAction = new SpriteAction() @@ -272,7 +273,213 @@ export default class InitCommand extends BaseCommand { const characterType = new CharacterType() await characterType .setId('75b70c78-17f0-44c0-a4fa-15043cb95be0') - .setName('New character type') + .setName('Male character') + .setGender(CharacterGender.MALE) + .setRace(CharacterRace.HUMAN) + .setIsSelectable(true) + .setSprite(characterSprite) + .save() + } + + private async createFemaleCharacterType(): Promise { + const characterSprite = new Sprite() + characterSprite.setId('023d1e9d-f57f-4faa-8412-86c07107cf85').setName('Male character') + await characterSprite.save() + + const idleRightDownAction = new SpriteAction() + await idleRightDownAction + .setAction('idle_right_down') + .setSprites([ + { + url: '', + offset: { + x: 0, + y: 0 + } + } + ]) + .setOriginX(0) + .setOriginY(0) + .setFrameWidth(28) + .setFrameHeight(94) + .setFrameRate(0) + .setSprite(characterSprite) + .save() + + const idleLeftUpAction = new SpriteAction() + await idleLeftUpAction + .setAction('idle_left_up') + .setSprites([ + { + url: '', + offset: { + x: 0, + y: 0 + } + } + ]) + .setOriginX(0) + .setOriginY(0) + .setFrameWidth(26) + .setFrameHeight(93) + .setFrameRate(0) + .setSprite(characterSprite) + .save() + + const walkRightDownAction = new SpriteAction() + await walkRightDownAction + .setAction('walk_right_down') + .setSprites([ + { + url: '', + offset: { + x: 7, + y: 8 + } + }, + { + url: '', + offset: { + x: 7, + y: 2 + } + }, + { + url: '', + offset: { + x: 2, + y: 2 + } + }, + { + url: '', + offset: { + x: 0, + y: 0 + } + } + ]) + .setOriginX(0) + .setOriginY(0) + .setFrameWidth(36) + .setFrameHeight(102) + .setFrameRate(7) + .setSprite(characterSprite) + .save() + + const walkLeftUpAction = new SpriteAction() + await walkLeftUpAction + .setAction('walk_left_up') + .setSprites([ + { + url: '', + offset: { + x: 3, + y: 2 + } + }, + { + url: '', + offset: { + x: 0, + y: 2 + } + }, + { + url: '', + offset: { + x: 5, + y: 6 + } + }, + { + url: '', + offset: { + x: 2, + y: 6 + } + } + ]) + .setOriginX(0) + .setOriginY(0) + .setFrameWidth(34) + .setFrameHeight(101) + .setFrameRate(7) + .setSprite(characterSprite) + .save() + + const attackRightDownAction = new SpriteAction() + await attackRightDownAction + .setAction('attack_right_down') + .setSprites([ + { + url: '', + offset: { + x: 20, + y: 0 + } + }, + { + url: '', + offset: { + x: 19, + y: 8 + } + }, + { + url: '', + offset: { + x: 17, + y: 3 + } + } + ]) + .setOriginX(0) + .setOriginY(0) + .setFrameWidth(69) + .setFrameHeight(111) + .setFrameRate(5) + .setSprite(characterSprite) + .save() + + const attackLeftUpAction = new SpriteAction() + await attackLeftUpAction + .setAction('attack_left_up') + .setSprites([ + { + url: '', + offset: { + x: 2, + y: 0 + } + }, + { + url: '', + offset: { + x: 5, + y: 0 + } + }, + { + url: '', + offset: { + x: 6, + y: 1 + } + } + ]) + .setOriginX(0) + .setOriginY(0) + .setFrameWidth(34) + .setFrameHeight(100) + .setFrameRate(5) + .setSprite(characterSprite) + .save() + + const characterType = new CharacterType() + await characterType + .setId('75b70c78-17f0-44c0-a4fa-15043cb95be0') + .setName('Male character') .setGender(CharacterGender.MALE) .setRace(CharacterRace.HUMAN) .setIsSelectable(true) @@ -282,7 +489,7 @@ export default class InitCommand extends BaseCommand { private async createCharacterHair(): Promise { const hairSprite = new Sprite() - hairSprite.setId('922ee95f-1500-49c0-8ead-f8cc46dad136').setName('Hair 1') + hairSprite.setId('922ee95f-1500-49c0-8ead-f8cc46dad136').setName('Hair 1').setWidth(30).setHeight(40) await hairSprite.save() const frontAction = new SpriteAction() @@ -326,7 +533,7 @@ export default class InitCommand extends BaseCommand { .save() const characterHair = new CharacterHair() - await characterHair.setId('a2471230-d238-4ffb-9eca-9eab869f1b67').setName('Hair 1').setGender(CharacterGender.MALE).setIsSelectable(true).setSprite(hairSprite).save() + await characterHair.setId('a2471230-d238-4ffb-9eca-9eab869f1b67').setName('Hair 1').setGender(CharacterGender.MALE).setColor('#1B1212').setIsSelectable(true).setSprite(hairSprite).save() } private async createMap(): Promise { @@ -345,6 +552,8 @@ export default class InitCommand extends BaseCommand { private async createUser(): Promise { const user = new User() await user.setId('6f9a58b4-172d-425e-b9ea-71e1d13d81ee').setUsername('root').setEmail('local@host').setPassword('password').setOnline(false).save() + const map = await this.mapRepository.getFirst() + if (!map) return const character = new Character() await character @@ -352,7 +561,7 @@ export default class InitCommand extends BaseCommand { .setUser(user) .setName('root') .setRole('gm') - .setMap(await this.mapRepository.getFirst()) + .setMap(map) .setCharacterType(await this.characterTypeRepository.getFirst()) .setCharacterHair(await this.characterHairRepository.getFirst()) .save() diff --git a/src/controllers/avatar.ts b/src/controllers/avatar.ts index 803d607..7256a47 100644 --- a/src/controllers/avatar.ts +++ b/src/controllers/avatar.ts @@ -65,9 +65,12 @@ export class AvatarController extends BaseController { return this.sendError(res, 'Body sprite file not found', 404) } + // Get body sprite metadata + const bodyMetadata = await sharp(bodySpritePath).metadata() + let avatar = sharp(bodySpritePath).extend({ - top: 2, - bottom: 2, + top: 0, + bottom: 0, background: { r: 0, g: 0, b: 0, alpha: 0 } }) @@ -76,7 +79,21 @@ export class AvatarController extends BaseController { if (characterHair?.sprite?.id) { const hairSpritePath = Storage.getPublicPath('sprites', characterHair.sprite.id, 'front.png') if (fs.existsSync(hairSpritePath)) { - avatar = avatar.composite([{ input: hairSpritePath, gravity: 'north' }]) + // Resize hair sprite to match body dimensions + const resizedHair = await sharp(hairSpritePath) + .resize(bodyMetadata.width, bodyMetadata.height, { + fit: 'contain', + background: { r: 0, g: 0, b: 0, alpha: 0 } + }) + .toBuffer() + + avatar = avatar.composite([ + { + input: resizedHair, + left: 0, + top: -27 // Apply vertical offset + } + ]) } } } @@ -84,6 +101,7 @@ export class AvatarController extends BaseController { res.setHeader('Content-Type', 'image/png') return avatar.pipe(res) } catch (error) { + console.error('Avatar generation error:', error) return this.sendError(res, 'Error generating avatar', 500) } } diff --git a/src/entities/base/characterHair.ts b/src/entities/base/characterHair.ts index b257cae..51f801f 100644 --- a/src/entities/base/characterHair.ts +++ b/src/entities/base/characterHair.ts @@ -16,11 +16,14 @@ export class BaseCharacterHair extends BaseEntity { @Property() gender: CharacterGender = CharacterGender.MALE + @Property() + color: string = '#000000' + @Property() isSelectable = false @ManyToOne() - sprite?: Sprite + sprite!: Sprite @Property() createdAt = new Date() @@ -55,6 +58,15 @@ export class BaseCharacterHair extends BaseEntity { return this.gender } + setColor(color: string) { + this.color = color + return this + } + + getColor() { + return this.color + } + setIsSelectable(isSelectable: boolean) { this.isSelectable = isSelectable return this diff --git a/src/entities/base/item.ts b/src/entities/base/item.ts index 8e837b7..01b2808 100644 --- a/src/entities/base/item.ts +++ b/src/entities/base/item.ts @@ -25,8 +25,8 @@ export class BaseItem extends BaseEntity { @Enum(() => ItemRarity) rarity: ItemRarity = ItemRarity.COMMON - @ManyToOne(() => Sprite) - sprite?: Sprite + @ManyToOne() + sprite!: Sprite @Property() createdAt = new Date() diff --git a/src/entities/base/map.ts b/src/entities/base/map.ts index d40431b..fbda149 100644 --- a/src/entities/base/map.ts +++ b/src/entities/base/map.ts @@ -11,7 +11,7 @@ export class BaseMap extends BaseEntity { id = randomUUID() @Property() - name!: string + name: string = '' @Property() width = 10 @@ -19,7 +19,7 @@ export class BaseMap extends BaseEntity { @Property() height = 10 - @Property({ type: 'json', nullable: true }) + @Property({ type: 'json' }) tiles: Array> = [] @Property() diff --git a/src/entities/base/mapEventTile.ts b/src/entities/base/mapEventTile.ts index afc8e70..4f75ffe 100644 --- a/src/entities/base/mapEventTile.ts +++ b/src/entities/base/mapEventTile.ts @@ -17,12 +17,12 @@ export class BaseMapEventTile extends BaseEntity { type!: MapEventTileType @Property() - positionX!: number + positionX: number = 0 @Property() - positionY!: number + positionY: number = 0 - @OneToOne({ eager: true }) + @OneToOne({ eager: true, deleteRule: 'cascade', orphanRemoval: true }) teleport?: MapEventTileTeleport setId(id: UUID) { diff --git a/src/entities/base/mapEventTileTeleport.ts b/src/entities/base/mapEventTileTeleport.ts index d330459..f156d6c 100644 --- a/src/entities/base/mapEventTileTeleport.ts +++ b/src/entities/base/mapEventTileTeleport.ts @@ -9,7 +9,7 @@ export class BaseMapEventTileTeleport extends BaseEntity { @PrimaryKey() id = randomUUID() - @OneToOne({ deleteRule: 'cascade' }) + @OneToOne({ deleteRule: 'cascade', orphanRemoval: true }) mapEventTile!: MapEventTile @ManyToOne({ deleteRule: 'cascade', eager: true }) diff --git a/src/entities/base/mapObject.ts b/src/entities/base/mapObject.ts index d956be3..cd61d38 100644 --- a/src/entities/base/mapObject.ts +++ b/src/entities/base/mapObject.ts @@ -16,6 +16,9 @@ export class BaseMapObject extends BaseEntity { @Property({ type: 'json' }) depthOffsets = [0] + @Property({ type: 'json' }) + pivotPoints: { x: number; y: number }[] = [] + @Property({ type: 'decimal', precision: 10, scale: 2 }) originX = 0 @@ -66,6 +69,10 @@ export class BaseMapObject extends BaseEntity { setDepthOffsets(offsets: any) { this.depthOffsets = offsets + } + + setPivotPoints(pivotPoints: { x: number; y: number }[]) { + this.pivotPoints = pivotPoints return this } diff --git a/src/entities/base/sprite.ts b/src/entities/base/sprite.ts index 2ef9e54..874690a 100644 --- a/src/entities/base/sprite.ts +++ b/src/entities/base/sprite.ts @@ -11,9 +11,15 @@ export class BaseSprite extends BaseEntity { @Property() name!: string - @OneToMany(() => SpriteAction, (action) => action.sprite) + @OneToMany({ mappedBy: 'sprite', orphanRemoval: true }) spriteActions = new Collection(this) + @Property({ nullable: true }) + width: number | null = null + + @Property({ nullable: true }) + height: number | null = null + @Property() createdAt = new Date() @@ -47,6 +53,24 @@ export class BaseSprite extends BaseEntity { return this.spriteActions } + setWidth(width: number | null) { + this.width = width + return this + } + + getWidth() { + return this.width + } + + setHeight(height: number | null) { + this.height = height + return this + } + + getHeight() { + return this.height + } + setCreatedAt(createdAt: Date) { this.createdAt = createdAt return this diff --git a/src/entities/map.ts b/src/entities/map.ts index e7f6f4e..cdb3d29 100644 --- a/src/entities/map.ts +++ b/src/entities/map.ts @@ -2,6 +2,7 @@ import { BaseMap } from '@/entities/base/map' import { Entity } from '@mikro-orm/core' export type MapCacheT = ReturnType | {} +export type MapEditorMapT = ReturnType | {} @Entity() export class Map extends BaseMap { @@ -35,4 +36,42 @@ export class Map extends BaseMap { return {} } } + + public async mapEditorObject() { + try { + await this.getPlacedMapObjects().load() + await this.getMapEffects().load() + await this.getMapEventTiles().load() + + return { + id: this.getId(), + name: this.getName(), + width: this.getWidth(), + height: this.getHeight(), + tiles: this.getTiles(), + pvp: this.getPvp(), + createdAt: this.getCreatedAt(), + updatedAt: this.getUpdatedAt(), + placedMapObjects: this.getPlacedMapObjects(), + mapEffects: this.getMapEffects(), + mapEventTiles: this.getMapEventTiles().map((mapEventTile) => ({ + id: mapEventTile.getId(), + type: mapEventTile.getType(), + positionX: mapEventTile.getPositionX(), + positionY: mapEventTile.getPositionY(), + teleport: mapEventTile.getTeleport() + ? { + toMap: mapEventTile.getTeleport()?.getToMap().getId(), + toPositionX: mapEventTile.getTeleport()?.getToPositionX(), + toPositionY: mapEventTile.getTeleport()?.getToPositionY(), + toRotation: mapEventTile.getTeleport()?.getToRotation() + } + : undefined + })) + } + } catch (error) { + console.error(error) + return {} + } + } } diff --git a/src/events/character/connect.ts b/src/events/character/connect.ts index 49c984e..fb7c1ed 100644 --- a/src/events/character/connect.ts +++ b/src/events/character/connect.ts @@ -1,6 +1,7 @@ import { BaseEvent } from '@/application/base/baseEvent' import { SocketEvent } from '@/application/enums' import type { UUID } from '@/application/types' +import { ZCharacterConnect } from '@/application/zodTypes' import MapManager from '@/managers/mapManager' import CharacterHairRepository from '@/repositories/characterHairRepository' import CharacterRepository from '@/repositories/characterRepository' @@ -8,7 +9,8 @@ import TeleportService from '@/services/characterTeleportService' interface CharacterConnectPayload { characterId: UUID - characterHairId?: UUID + characterHairId: UUID | null + newNickname?: string } export default class CharacterConnectEvent extends BaseEvent { @@ -21,18 +23,35 @@ export default class CharacterConnectEvent extends BaseEvent { private async handleEvent(data: CharacterConnectPayload, callback: (response: any) => void): Promise { try { - if (await this.checkForActiveCharacters()) { - this.emitError('You are already connected to another character') + if (data.newNickname === '') data.newNickname = undefined + const result = ZCharacterConnect.safeParse(data) + if (!result.success) { + this.sendNotificationAndLog(result.error?.errors[0]?.message ?? 'Invalid data') return } - const character = await this.characterRepository.getByUserAndId(this.socket.userId, data.characterId) + if (await this.checkForActiveCharacters()) { + this.sendNotificationAndLog('You are already connected to another character') + return + } + + let character = await this.characterRepository.getByUserAndId(this.socket.userId!, data.characterId) if (!character) { - this.emitError('Character not found or does not belong to this user') + this.sendNotificationAndLog('Character not found or does not belong to this user') return } + if (data.newNickname) { + const existingCharacter = await this.characterRepository.getByName(data.newNickname) + if (existingCharacter) { + this.sendNotificationAndLog('Nickname already in use: ' + data.newNickname) + return + } + + await character.setName(data.newNickname).save() + } + // Set character id this.socket.characterId = character.id @@ -40,6 +59,8 @@ export default class CharacterConnectEvent extends BaseEvent { if (data.characterHairId !== undefined && data.characterHairId !== null) { const characterHair = await this.characterHairRepository.getById(data.characterHairId) await character.setCharacterHair(characterHair).save() + } else { + await character.setCharacterHair(null).save() } // Emit character connect event @@ -62,7 +83,7 @@ export default class CharacterConnectEvent extends BaseEvent { } private async checkForActiveCharacters(): Promise { - const characters = await this.characterRepository.getByUserId(this.socket.userId) + const characters = await this.characterRepository.getByUserId(this.socket.userId!) return characters?.some((char) => MapManager.getCharacterById(char.id)) ?? false } } diff --git a/src/events/character/create.ts b/src/events/character/create.ts index e4e9a42..1895c37 100644 --- a/src/events/character/create.ts +++ b/src/events/character/create.ts @@ -32,7 +32,7 @@ export default class CharacterCreateEvent extends BaseEvent { } private async createCharacter(data: z.infer): Promise { - const user = await this.userRepository.getById(this.socket.userId) + const user = await this.userRepository.getById(this.socket.userId!) if (!user) { throw new Error('You are not logged in') } diff --git a/src/events/disconnect.ts b/src/events/disconnect.ts index f570ce3..3ba5205 100644 --- a/src/events/disconnect.ts +++ b/src/events/disconnect.ts @@ -4,7 +4,7 @@ import MapManager from '@/managers/mapManager' export default class DisconnectEvent extends BaseEvent { public listen(): void { - this.socket.on('disconnect', this.handleEvent.bind(this)) + this.socket.on(SocketEvent.DISCONNECT, this.handleEvent.bind(this)) } private async handleEvent(): Promise { diff --git a/src/events/gameMaster/assetManager/characterHair/create.ts b/src/events/gameMaster/assetManager/characterHair/create.ts index 6952ea4..66977a2 100644 --- a/src/events/gameMaster/assetManager/characterHair/create.ts +++ b/src/events/gameMaster/assetManager/characterHair/create.ts @@ -1,6 +1,7 @@ import { BaseEvent } from '@/application/base/baseEvent' -import { SocketEvent } from '@/application/enums' +import { CharacterGender, SocketEvent } from '@/application/enums' import { CharacterHair } from '@/entities/characterHair' +import SpriteRepository from '@/repositories/spriteRepository' export default class CharacterHairCreateEvent extends BaseEvent { public listen(): void { @@ -11,8 +12,17 @@ export default class CharacterHairCreateEvent extends BaseEvent { try { if (!(await this.isCharacterGM())) return + // Get first sprite + const spriteRepository = new SpriteRepository() + const firstSprite = await spriteRepository.getFirst() + + if (!firstSprite) { + this.sendNotificationAndLog('No sprites found') + return callback(false) + } + const newCharacterHair = new CharacterHair() - await newCharacterHair.setName('New hair').save() + await newCharacterHair.setName('New hair').setGender(CharacterGender.MALE).setSprite(firstSprite).save() return callback(true) } catch (error) { diff --git a/src/events/gameMaster/assetManager/characterHair/update.ts b/src/events/gameMaster/assetManager/characterHair/update.ts index 8b03457..b2711d4 100644 --- a/src/events/gameMaster/assetManager/characterHair/update.ts +++ b/src/events/gameMaster/assetManager/characterHair/update.ts @@ -8,6 +8,7 @@ type Payload = { id: UUID name: string gender: CharacterGender + color: string isSelectable: boolean spriteId: UUID } @@ -29,7 +30,7 @@ export default class CharacterHairUpdateEvent extends BaseEvent { const characterHair = await characterHairRepository.getById(data.id) if (!characterHair) return callback(false) - await characterHair.setName(data.name).setGender(data.gender).setIsSelectable(data.isSelectable).setSprite(sprite).setUpdatedAt(new Date()).save() + await characterHair.setName(data.name).setGender(data.gender).setColor(data.color).setIsSelectable(data.isSelectable).setSprite(sprite).setUpdatedAt(new Date()).save() return callback(true) } catch (error) { diff --git a/src/events/gameMaster/assetManager/item/create.ts b/src/events/gameMaster/assetManager/item/create.ts index 3bad4d5..cd7cd83 100644 --- a/src/events/gameMaster/assetManager/item/create.ts +++ b/src/events/gameMaster/assetManager/item/create.ts @@ -14,7 +14,10 @@ export default class ItemCreateEvent extends BaseEvent { const spriteRepository = new SpriteRepository() const sprite = await spriteRepository.getFirst() - if (!sprite) return callback(false) + if (!sprite) { + this.sendNotificationAndLog('No sprites found') + return callback(false) + } const newItem = new Item() await newItem.setName('New Item').setItemType(ItemType.WEAPON).setStackable(false).setRarity(ItemRarity.COMMON).setSprite(sprite).save() diff --git a/src/events/gameMaster/assetManager/mapObject/update.ts b/src/events/gameMaster/assetManager/mapObject/update.ts index 6ad0276..e38e743 100644 --- a/src/events/gameMaster/assetManager/mapObject/update.ts +++ b/src/events/gameMaster/assetManager/mapObject/update.ts @@ -8,6 +8,7 @@ type Payload = { name: string tags: string[] depthOffsets: number[] + pivotPoints: { x: number; y: number }[] originX: number originY: number frameRate: number @@ -32,6 +33,7 @@ export default class MapObjectUpdateEvent extends BaseEvent { if (data.name !== undefined) mapObject.name = data.name if (data.tags !== undefined) mapObject.tags = data.tags if (data.depthOffsets !== undefined) mapObject.depthOffsets = data.depthOffsets + if (data.pivotPoints !== undefined) mapObject.pivotPoints = data.pivotPoints if (data.originX !== undefined) mapObject.originX = data.originX if (data.originY !== undefined) mapObject.originY = data.originY if (data.frameRate !== undefined) mapObject.frameRate = data.frameRate diff --git a/src/events/gameMaster/assetManager/sprite/copy.ts b/src/events/gameMaster/assetManager/sprite/copy.ts index 128a083..c38d8fb 100644 --- a/src/events/gameMaster/assetManager/sprite/copy.ts +++ b/src/events/gameMaster/assetManager/sprite/copy.ts @@ -2,6 +2,7 @@ import { BaseEvent } from '@/application/base/baseEvent' import { SocketEvent } from '@/application/enums' import type { UUID } from '@/application/types' import { Sprite } from '@/entities/sprite' +import { SpriteAction } from '@/entities/spriteAction' import SpriteRepository from '@/repositories/spriteRepository' interface CopyPayload { @@ -29,7 +30,21 @@ export default class SpriteCopyEvent extends BaseEvent { await spriteRepository.getEntityManager().populate(sourceSprite, ['spriteActions']) const newSprite = new Sprite() - await newSprite.setName(`${sourceSprite.getName()} (Copy)`).setSpriteActions(sourceSprite.getSpriteActions()).save() + await newSprite.setName(`${sourceSprite.getName()} (Copy)`).save() + + for (const spriteAction of sourceSprite.getSpriteActions()) { + const newSpriteAction = new SpriteAction() + await newSpriteAction + .setSprite(newSprite) + .setAction(spriteAction.getAction()) + .setSprites(spriteAction.getSprites() ?? []) + .setOriginX(spriteAction.getOriginX()) + .setOriginY(spriteAction.getOriginY()) + .setFrameWidth(spriteAction.getFrameWidth()) + .setFrameHeight(spriteAction.getFrameHeight()) + .setFrameRate(spriteAction.getFrameRate()) + .save() + } return callback(true) } catch (error) { diff --git a/src/events/gameMaster/assetManager/sprite/update.ts b/src/events/gameMaster/assetManager/sprite/update.ts index 7bac5cf..4829b84 100644 --- a/src/events/gameMaster/assetManager/sprite/update.ts +++ b/src/events/gameMaster/assetManager/sprite/update.ts @@ -14,76 +14,103 @@ interface SpriteImage { } } -interface ImageDimensions { - width: number - height: number - offsetX: number - offsetY: number +interface SpriteActionData { + action: string + sprites: SpriteImage[] + originX: number + originY: number + frameRate: number } -interface EffectiveDimensions { - width: number - height: number - top: number - bottom: number -} - -type Payload = { +interface UpdateSpritePayload { id: UUID name: string - spriteActions: Array<{ - action: string - sprites: SpriteImage[] - originX: number - originY: number - frameRate: number - }> + spriteActions: SpriteActionData[] } export default class SpriteUpdateEvent extends BaseEvent { + private readonly spriteRepository: SpriteRepository = new SpriteRepository() + public listen(): void { this.socket.on(SocketEvent.GM_SPRITE_UPDATE, this.handleEvent.bind(this)) } - private async handleEvent(data: Payload, callback: (success: boolean) => void): Promise { + private async handleEvent(data: UpdateSpritePayload, callback: (success: boolean) => void): Promise { try { - if (!(await this.isCharacterGM())) return - - const spriteRepository = new SpriteRepository() - const sprite = await spriteRepository.getById(data.id) - if (!sprite) return callback(false) - - await spriteRepository.getEntityManager().populate(sprite, ['spriteActions']) - - // Update sprite in database - await sprite.setName(data.name).save() - - // First verify all sprite sheets can be generated - for (const actionData of data.spriteActions) { - if (!(await this.generateSpriteSheet(actionData.sprites, sprite.getId(), actionData.action))) { - return callback(false) - } + // Validate request and permissions + if (!(await this.isCharacterGM())) { + callback(false) + return } + // Get and validate sprite + const sprite = await this.spriteRepository.getById(data.id) + if (!sprite) { + callback(false) + return + } + await this.spriteRepository.getEntityManager().populate(sprite, ['spriteActions']) + + // Update sprite name + await sprite.setName(data.name).setUpdatedAt(new Date()).save() + + // Process all sprite actions + const success = await this.processAllSpriteActions(data.spriteActions, sprite) + callback(success) + } catch (error) { + console.error(`Error updating sprite ${data.id}:`, error) + callback(false) + } + } + + private async processAllSpriteActions(actionsData: SpriteActionData[], sprite: any): Promise { + try { + // Remove existing actions const existingActions = sprite.getSpriteActions() - - // Remove existing actions only after confirming sprite sheets generated successfully for (const existingAction of existingActions) { - await spriteRepository.getEntityManager().removeAndFlush(existingAction) + await this.spriteRepository.getEntityManager().removeAndFlush(existingAction) } - // Create new actions - for (const actionData of data.spriteActions) { - // Process images and calculate dimensions - const imageData = await Promise.all(actionData.sprites.map((sprite) => this.processImage(sprite))) - const effectiveDimensions = imageData.map((dimensions) => this.calculateEffectiveDimensions(dimensions)) + // First pass: find the global maximum dimensions across all actions + let globalMaxWidth = 0 + let globalMaxHeight = 0 - // Calculate total height needed for the sprite sheet - const maxHeight = Math.max(...effectiveDimensions.map((d) => d.height)) - const maxTop = Math.max(...effectiveDimensions.map((d) => d.top)) - const maxBottom = Math.max(...effectiveDimensions.map((d) => d.bottom)) - const totalHeight = maxHeight + maxTop + maxBottom + // Extract all image metadata to find global maximums + for (const actionData of actionsData) { + if (actionData.sprites.length === 0) continue + const imagesData = await Promise.all( + actionData.sprites.map(async (sprite) => { + const base64Data = sprite.url.split(';base64,').pop() + if (!base64Data) throw new Error('Invalid base64 image') + const buffer = Buffer.from(base64Data, 'base64') + const metadata = await sharp(buffer).metadata() + + return { + width: metadata.width || 0, + height: metadata.height || 0 + } + }) + ) + + // Update global maximums with this action's maximums + const actionMaxWidth = Math.max(...imagesData.map((data) => data.width), 0) + const actionMaxHeight = Math.max(...imagesData.map((data) => data.height), 0) + + globalMaxWidth = Math.max(globalMaxWidth, actionMaxWidth) + globalMaxHeight = Math.max(globalMaxHeight, actionMaxHeight) + } + + // Process each action using the global maximum dimensions + for (const actionData of actionsData) { + if (actionData.sprites.length === 0) continue + + // Generate and save the sprite sheet using global dimensions + const frameDimensions = await this.generateAndSaveSpriteSheet(actionData.sprites, sprite.getId(), actionData.action, globalMaxWidth, globalMaxHeight) + + if (!frameDimensions) return false + + // Create and save sprite action const spriteAction = new SpriteAction() spriteAction.setSprite(sprite) sprite.getSpriteActions().add(spriteAction) @@ -93,73 +120,85 @@ export default class SpriteUpdateEvent extends BaseEvent { .setSprites(actionData.sprites) .setOriginX(actionData.originX) .setOriginY(actionData.originY) - .setFrameWidth(await this.calculateMaxWidth(actionData.sprites)) - .setFrameHeight(totalHeight) + .setFrameWidth(frameDimensions.frameWidth) + .setFrameHeight(frameDimensions.frameHeight) .setFrameRate(actionData.frameRate) - await spriteRepository.getEntityManager().persistAndFlush(spriteAction) + await this.spriteRepository.getEntityManager().persistAndFlush(spriteAction) } - return callback(true) + return true } catch (error) { - console.error(`Error updating sprite ${data.id}:`, error) - return callback(false) + console.error('Error processing sprite actions:', error) + return false } } - private async generateSpriteSheet(sprites: SpriteImage[], spriteId: string, action: string): Promise { + private async generateAndSaveSpriteSheet(sprites: SpriteImage[], spriteId: string, action: string, maxWidth: number, maxHeight: number): Promise<{ frameWidth: number; frameHeight: number } | null> { try { - if (!sprites.length) return true + if (sprites.length === 0) return { frameWidth: 0, frameHeight: 0 } - // Process all images and get their dimensions - const imageData = await Promise.all(sprites.map((sprite) => this.processImage(sprite))) - const effectiveDimensions = imageData.map((dimensions) => this.calculateEffectiveDimensions(dimensions)) + // Extract image data + const imagesData = await Promise.all( + sprites.map(async (sprite) => { + const base64Data = sprite.url.split(';base64,').pop() + if (!base64Data) throw new Error('Invalid base64 image') + const buffer = Buffer.from(base64Data, 'base64') + const metadata = await sharp(buffer).metadata() - // Calculate maximum dimensions - const maxWidth = Math.max(...effectiveDimensions.map((d) => d.width)) - const maxHeight = Math.max(...effectiveDimensions.map((d) => d.height)) - const maxTop = Math.max(...effectiveDimensions.map((d) => d.top)) - const maxBottom = Math.max(...effectiveDimensions.map((d) => d.bottom)) + return { + buffer, + width: metadata.width || 0, + height: metadata.height || 0 + } + }) + ) - // Calculate total height needed - const totalHeight = maxHeight + maxTop + maxBottom + // Skip creation if any image has invalid dimensions + if (imagesData.some((data) => data.width === 0 || data.height === 0)) { + console.error('One or more sprites have invalid dimensions') + return null + } - // Process images and create sprite sheet - const processedImages = await Promise.all( - sprites.map(async (sprite, index) => { - const { width, height, offsetX, offsetY } = await this.processImage(sprite) - const uri = sprite.url.split(';base64,').pop() - if (!uri) throw new Error('Invalid base64 image') - const buffer = Buffer.from(uri, 'base64') + // Create frames of uniform size with the original sprites centered + const uniformFrames = await Promise.all( + imagesData.map(async (imageData) => { + // Calculate centering offsets to position the sprite in the middle of the frame + const xOffset = Math.floor((maxWidth - imageData.width) / 2) + const yOffset = Math.floor((maxHeight - imageData.height) / 2) - // Create individual frame - const left = offsetX >= 0 ? offsetX : 0 - const verticalOffset = totalHeight - height - (offsetY >= 0 ? offsetY : 0) + // Create a uniform-sized frame with the sprite centered return sharp({ create: { width: maxWidth, - height: totalHeight, + height: maxHeight, channels: 4, background: { r: 0, g: 0, b: 0, alpha: 0 } } }) - .composite([{ input: buffer, left, top: verticalOffset }]) + .composite([ + { + input: imageData.buffer, + left: xOffset, + top: yOffset + } + ]) .png() .toBuffer() }) ) - // Combine frames into sprite sheet + // Create the sprite sheet with uniform frames const spriteSheet = await sharp({ create: { - width: maxWidth * sprites.length, - height: totalHeight, + width: maxWidth * uniformFrames.length, + height: maxHeight, channels: 4, background: { r: 0, g: 0, b: 0, alpha: 0 } } }) .composite( - processedImages.map((buffer, index) => ({ + uniformFrames.map((buffer, index) => ({ input: buffer, left: index * maxWidth, top: 0 @@ -168,50 +207,19 @@ export default class SpriteUpdateEvent extends BaseEvent { .png() .toBuffer() - // Ensure directory exists + // Save sprite sheet const dir = `public/sprites/${spriteId}` await fs.promises.mkdir(dir, { recursive: true }) - - // Save the sprite sheet await fs.promises.writeFile(`${dir}/${action}.png`, spriteSheet) - return true + // Return the uniform frame dimensions (now global maximum dimensions) + return { + frameWidth: maxWidth, + frameHeight: maxHeight + } } catch (error) { console.error('Error generating sprite sheet:', error) - return false + return null } } - - private async processImage(sprite: SpriteImage): Promise { - const uri = sprite.url.split(';base64,').pop() - if (!uri) throw new Error('Invalid base64 image') - const buffer = Buffer.from(uri, 'base64') - const metadata = await sharp(buffer).metadata() - return { - width: metadata.width ?? 0, - height: metadata.height ?? 0, - offsetX: sprite.offset?.x ?? 0, - offsetY: sprite.offset?.y ?? 0 - } - } - - private calculateEffectiveDimensions(imageDimensions: ImageDimensions): EffectiveDimensions { - return { - width: imageDimensions.width + Math.abs(imageDimensions.offsetX), - height: imageDimensions.height + Math.abs(imageDimensions.offsetY), - top: imageDimensions.offsetY >= 0 ? imageDimensions.offsetY : 0, - bottom: imageDimensions.offsetY < 0 ? Math.abs(imageDimensions.offsetY) : 0 - } - } - - private async calculateMaxWidth(sprites: SpriteImage[]): Promise { - if (!sprites.length) return 0 - - // Process all images and get their dimensions - const imageData = await Promise.all(sprites.map((sprite) => this.processImage(sprite))) - const effectiveDimensions = imageData.map((dimensions) => this.calculateEffectiveDimensions(dimensions)) - - // Calculate maximum width needed - return Math.max(...effectiveDimensions.map((d) => d.width)) - } } diff --git a/src/events/gameMaster/mapEditor/request.ts b/src/events/gameMaster/mapEditor/request.ts index dc9fe93..c1885db 100644 --- a/src/events/gameMaster/mapEditor/request.ts +++ b/src/events/gameMaster/mapEditor/request.ts @@ -1,7 +1,7 @@ import { BaseEvent } from '@/application/base/baseEvent' import { SocketEvent } from '@/application/enums' import type { UUID } from '@/application/types' -import { Map } from '@/entities/map' +import { type MapEditorMapT } from '@/entities/map' import MapRepository from '@/repositories/mapRepository' interface IPayload { @@ -13,7 +13,7 @@ export default class MapRequestEvent extends BaseEvent { this.socket.on(SocketEvent.GM_MAP_REQUEST, this.handleEvent.bind(this)) } - private async handleEvent(data: IPayload, callback: (response: Map | null) => void): Promise { + private async handleEvent(data: IPayload, callback: (response: MapEditorMapT | null) => void): Promise { try { if (!(await this.isCharacterGM())) return @@ -34,9 +34,7 @@ export default class MapRequestEvent extends BaseEvent { await mapRepository.getEntityManager().populate(map, mapRepository.POPULATE_MAP_EDITOR as any) - // Remove map.mapEventTiles.teleport.toMap and add map.mapEventTiles.teleport.toMapId - - return callback(map) + return callback(await map.mapEditorObject()) } catch (error: any) { this.logger.error('gm:map:request error', error.message) return callback(null) diff --git a/src/events/gameMaster/mapEditor/update.ts b/src/events/gameMaster/mapEditor/update.ts index 1df390d..72c99b2 100644 --- a/src/events/gameMaster/mapEditor/update.ts +++ b/src/events/gameMaster/mapEditor/update.ts @@ -1,7 +1,7 @@ import { BaseEvent } from '@/application/base/baseEvent' import { MapEventTileType, SocketEvent } from '@/application/enums' import type { UUID } from '@/application/types' -import { Map } from '@/entities/map' +import { type MapEditorMapT } from '@/entities/map' import { MapEffect } from '@/entities/mapEffect' import { MapEventTile } from '@/entities/mapEventTile' import { MapEventTileTeleport } from '@/entities/mapEventTileTeleport' @@ -21,7 +21,7 @@ interface IPayload { positionX: number positionY: number teleport?: { - toMapId: string + toMap: string toPositionX: number toPositionY: number toRotation: number @@ -36,7 +36,7 @@ export default class MapUpdateEvent extends BaseEvent { this.socket.on(SocketEvent.GM_MAP_UPDATE, this.handleEvent.bind(this)) } - private async handleEvent(data: IPayload, callback: (response: Map | null) => void): Promise { + private async handleEvent(data: IPayload, callback: (response: MapEditorMapT | null) => void): Promise { try { if (!(await this.isCharacterGM())) return @@ -80,17 +80,17 @@ export default class MapUpdateEvent extends BaseEvent { map.getMapEffects().removeAll() // Create and add new map event tiles - for (const tile of data.mapEventTiles) { - const mapEventTile = new MapEventTile().setType(tile.type).setPositionX(tile.positionX).setPositionY(tile.positionY).setMap(map) - if (tile.teleport) { - const toMap = await mapRepository.getById(tile.teleport.toMapId as UUID) + for (const eventTile of data.mapEventTiles) { + const mapEventTile = new MapEventTile().setMap(map).setType(eventTile.type).setPositionX(eventTile.positionX).setPositionY(eventTile.positionY) + if (eventTile.teleport) { + const toMap = await mapRepository.getById(eventTile.teleport.toMap as UUID) if (!toMap) continue const teleport = new MapEventTileTeleport() .setMapEventTile(mapEventTile) .setToMap(toMap) - .setToPositionX(tile.teleport.toPositionX) - .setToPositionY(tile.teleport.toPositionY) - .setToRotation(tile.teleport.toRotation) + .setToPositionX(eventTile.teleport.toPositionX) + .setToPositionY(eventTile.teleport.toPositionY) + .setToRotation(eventTile.teleport.toRotation) mapEventTile.setTeleport(teleport) } @@ -128,9 +128,9 @@ export default class MapUpdateEvent extends BaseEvent { mapManager.unloadMap(data.mapId) await mapManager.loadMap(map) - return callback(map) + return callback(await map.mapEditorObject()) } catch (error: any) { - this.emitError(`gm:map:update error: ${error instanceof Error ? error.message + error.stack : String(error)}`) + this.sendNotificationAndLog(`gm:map:update error: ${error instanceof Error ? error.message + error.stack : String(error)}`) return callback(null) } } diff --git a/src/events/map/characterAttack.ts b/src/events/map/characterAttack.ts index 4ded236..8afbe7c 100644 --- a/src/events/map/characterAttack.ts +++ b/src/events/map/characterAttack.ts @@ -19,6 +19,11 @@ export default class CharacterMove extends BaseEvent { // Don't attack if the character is already moving if (this.getMapCharacter()?.isMoving) return + const throttleKey = `attack_${this.socket.characterId}` + if (this.isThrottled(throttleKey, 1000)) { + return + } + // Start attack await this.characterAttackService.attack(this.socket.characterId!) } catch (error) { diff --git a/src/events/map/characterMove.ts b/src/events/map/characterMove.ts index 633d127..c3ebf28 100644 --- a/src/events/map/characterMove.ts +++ b/src/events/map/characterMove.ts @@ -3,29 +3,24 @@ import { SocketEvent } from '@/application/enums' import type { MapEventTileWithTeleport } from '@/application/types' import MapManager from '@/managers/mapManager' import MapCharacter from '@/models/mapCharacter' -import MapEventTileRepository from '@/repositories/mapEventTileRepository' -import CharacterService from '@/services/characterMoveService' -import TeleportService from '@/services/characterTeleportService' +import CharacterMoveService from '@/services/characterMoveService' export default class CharacterMove extends BaseEvent { - private readonly characterService = CharacterService private readonly STEP_DELAY = 100 - private readonly THROTTLE_DELAY = 230 - private readonly MAX_REQUEST_DISTANCE = 30 // Maximum allowed distance for movement requests + private readonly THROTTLE_DELAY = 200 + private readonly MAX_REQUEST_DISTANCE = 30 private movementTimeout: NodeJS.Timeout | null = null private lastKnownPosition: { x: number; y: number } | null = null + private isProcessingMove = false public listen(): void { this.socket.on(SocketEvent.MAP_CHARACTER_MOVE, this.handleEvent.bind(this)) } - private async handleEvent({ positionX, positionY }: { positionX: number; positionY: number }): Promise { + private async handleEvent([positionX, positionY]: [number, number]): Promise { try { const mapCharacter = MapManager.getCharacterById(this.socket.characterId!) - if (!mapCharacter?.getCharacter()) { - this.logger.error('map:character:move error: Character not found or not initialized') - return - } + if (!mapCharacter?.getCharacter()) return const character = mapCharacter.getCharacter() const currentX = character.getPositionX() @@ -33,63 +28,56 @@ export default class CharacterMove extends BaseEvent { // Enhanced throttling with position tracking const throttleKey = `movement_${this.socket.characterId}` - if (this.isThrottled(throttleKey, this.THROTTLE_DELAY)) { - return - } + if (this.isThrottled(throttleKey, this.THROTTLE_DELAY)) return - // Validate current position against last known position - const movementValidation = this.characterService.validateMovementDistance(currentX, currentY, this.lastKnownPosition, this.STEP_DELAY, mapCharacter.isMoving) + // Stop any existing movement before starting a new one + await this.stopExistingMovement(mapCharacter) + + // Validate movement + const movementValidation = CharacterMoveService.validateMovementDistance(currentX, currentY, this.lastKnownPosition, this.STEP_DELAY, mapCharacter.isMoving) if (!movementValidation.isValid) { - this.logger.warn(`Suspicious movement detected: ${this.socket.characterId}`) - // Force position reset character.setPositionX(this.lastKnownPosition!.x).setPositionY(this.lastKnownPosition!.y) - this.broadcastMovement(character, false) + CharacterMoveService.broadcastMovement(character, false) return } - // Validate requested position distance - const requestValidation = this.characterService.validateRequestDistance(currentX, currentY, positionX, positionY, this.MAX_REQUEST_DISTANCE) - - if (!requestValidation.isValid) { - this.logger.warn(`Invalid movement distance detected: ${this.socket.characterId}`) - return - } - - // If character is already moving to the same target position, ignore the request - if (mapCharacter.isMoving && mapCharacter.currentPath?.length) { - const lastPathPoint = mapCharacter.currentPath[mapCharacter.currentPath.length - 1] - if (lastPathPoint && Math.abs(lastPathPoint.positionX - positionX) < 1 && Math.abs(lastPathPoint.positionY - positionY) < 1) { - return - } - } - - // Cancel any ongoing movement - this.cancelCurrentMovement(mapCharacter) - - // Update last known position - this.lastKnownPosition = { x: currentX, y: currentY } - - // Calculate path to target position - const path = await this.characterService.calculatePath(mapCharacter.character, Math.floor(positionX), Math.floor(positionY)) + // Calculate and validate path + const path = await CharacterMoveService.calculatePath(character, Math.floor(positionX), Math.floor(positionY)) if (!path?.length) { - this.io.in(mapCharacter.character.map.id).emit(SocketEvent.MAP_CHARACTER_MOVEERROR, 'No valid path found') + mapCharacter.isMoving = false + mapCharacter.currentPath = null + CharacterMoveService.broadcastMovement(character, false) return } - // Start new movement - mapCharacter.isMoving = true + // Start movement mapCharacter.currentPath = path + mapCharacter.isMoving = true + this.isProcessingMove = true + await this.moveAlongPath(mapCharacter) } catch (error: any) { this.logger.error('map:character:move error: ' + error.message) + } finally { + this.isProcessingMove = false } } - private cancelCurrentMovement(mapCharacter: MapCharacter): void { - if (!mapCharacter.isMoving) return - mapCharacter.isMoving = false + private async stopExistingMovement(mapCharacter: MapCharacter): Promise { + // Clear existing movement timeout + if (this.movementTimeout) { + clearTimeout(this.movementTimeout) + this.movementTimeout = null + } + + // Wait for any ongoing movement processing to complete + if (this.isProcessingMove) { + await new Promise((resolve) => setTimeout(resolve, this.STEP_DELAY)) + } + + // Only clear the path, keep isMoving state for continuous animation mapCharacter.currentPath = null } @@ -97,18 +85,18 @@ export default class CharacterMove extends BaseEvent { const character = mapCharacter.getCharacter() const path = mapCharacter.currentPath - if (!path?.length) return + if (!path?.length || !character) return let lastMoveTime = Date.now() let currentTile, nextTile try { for (let i = 0; i < path.length - 1; i++) { + // Check if this movement sequence is still valid if (!mapCharacter.isMoving || mapCharacter.currentPath !== path) { return } - // Ensure minimum time between moves using a single Date.now() call const timeSinceLastMove = Date.now() - lastMoveTime if (timeSinceLastMove < this.STEP_DELAY) { await new Promise((resolve) => setTimeout(resolve, this.STEP_DELAY - timeSinceLastMove)) @@ -117,36 +105,41 @@ export default class CharacterMove extends BaseEvent { currentTile = path[i] nextTile = path[i + 1] - if (!currentTile || !nextTile || !this.isValidStep(currentTile, nextTile)) { - this.logger.error('Invalid movement step detected') - break + if (!currentTile || !nextTile || !CharacterMoveService.isValidStep(currentTile, nextTile)) { + return } - // Update character rotation and position in a single operation + // Update character position and rotation character - .setRotation(CharacterService.calculateRotation(currentTile.positionX, currentTile.positionY, nextTile.positionX, nextTile.positionY)) + .setRotation(CharacterMoveService.calculateRotation(currentTile.positionX, currentTile.positionY, nextTile.positionX, nextTile.positionY)) .setPositionX(nextTile.positionX) .setPositionY(nextTile.positionY) - // Check for map events at the next tile - const mapEventTile = await this.checkMapEvents(character, nextTile) + // Check for map events + const mapEventTile = await CharacterMoveService.checkMapEvents(character.getMap().getId(), nextTile) + if (mapEventTile) { if (mapEventTile.type === 'BLOCK') break if (mapEventTile.type === 'TELEPORT' && mapEventTile.teleport) { - await this.handleTeleportMapEventTile(mapEventTile as MapEventTileWithTeleport) + // Force clear movement state before teleport + mapCharacter.isMoving = false + mapCharacter.currentPath = null + this.lastKnownPosition = null // Reset last known position + + await CharacterMoveService.handleTeleportMapEventTile(character.id, mapEventTile as MapEventTileWithTeleport) return } } - // Broadcast movement - this.broadcastMovement(character, true) + // Only broadcast if this is still the current movement + if (mapCharacter.isMoving && mapCharacter.currentPath === path) { + CharacterMoveService.broadcastMovement(character, true) + } - // Apply movement delay between steps if (i < path.length - 2) { await new Promise((resolve) => setTimeout(resolve, this.STEP_DELAY)) } - // Update last known position and move time this.lastKnownPosition = { x: nextTile.positionX, y: nextTile.positionY @@ -154,62 +147,31 @@ export default class CharacterMove extends BaseEvent { lastMoveTime = Date.now() } } finally { + // Only finalize if this movement is still active if (mapCharacter.isMoving && mapCharacter.currentPath === path) { - this.finalizeMovement(mapCharacter) + await this.finalizeMovement(mapCharacter) } } } - private isValidStep(current: { positionX: number; positionY: number }, next: { positionX: number; positionY: number }): boolean { - return Math.abs(next.positionX - current.positionX) <= 1 && Math.abs(next.positionY - current.positionY) <= 1 - } - - private async checkMapEvents(character: any, nextTile: { positionX: number; positionY: number }) { - const mapEventTileRepository = new MapEventTileRepository() - return mapEventTileRepository.getEventTileByMapIdAndPosition(character.getMap().getId(), Math.floor(nextTile.positionX), Math.floor(nextTile.positionY)) - } - - private async handleTeleportMapEventTile(mapEventTile: MapEventTileWithTeleport): Promise { - const teleport = mapEventTile.getTeleport() - if (teleport) { - await TeleportService.teleportCharacter(this.socket.characterId!, { - targetMapId: teleport.getToMap().getId(), - targetX: teleport.getToPositionX(), - targetY: teleport.getToPositionY(), - rotation: teleport.getToRotation() - }) - } - } - - private broadcastMovement(character: any, isMoving: boolean): void { - this.io.in(character.map.id).emit(SocketEvent.MAP_CHARACTER_MOVE, { - characterId: character.id, - positionX: character.getPositionX(), - positionY: character.getPositionY(), - rotation: character.getRotation(), - isMoving - }) - } - - private finalizeMovement(mapCharacter: MapCharacter): void { - // Clear any existing timeout + private async finalizeMovement(mapCharacter: MapCharacter): Promise { if (this.movementTimeout) { clearTimeout(this.movementTimeout) } - // Clear the current path immediately mapCharacter.currentPath = null - // Set new timeout for movement state cleanup - this.movementTimeout = setTimeout(() => { - // Ensure the character is still in a valid state - if (!mapCharacter.isMoving || !mapCharacter.currentPath) { - mapCharacter.isMoving = false - // Save the final position and broadcast it - mapCharacter.savePosition().then(() => { - this.broadcastMovement(mapCharacter.character, false) - }) - } - }, this.STEP_DELAY * 2) // Increased delay to ensure all movement processing is complete + const character = mapCharacter.getCharacter() + if (character) { + await mapCharacter.savePosition() + + // Set a timeout to stop movement animation if no new movement is received + this.movementTimeout = setTimeout(() => { + if (!mapCharacter.currentPath) { + mapCharacter.isMoving = false + CharacterMoveService.broadcastMovement(character, false) + } + }, this.THROTTLE_DELAY) // Use THROTTLE_DELAY to match input timing + } } } diff --git a/src/managers/mapManager.ts b/src/managers/mapManager.ts index 23705fa..4f846a8 100644 --- a/src/managers/mapManager.ts +++ b/src/managers/mapManager.ts @@ -15,16 +15,20 @@ class MapManager { await mapRepository.getEntityManager().populate(maps, mapRepository.POPULATE_ALL as never[]) await Promise.all(maps.map((map) => this.loadMap(map))) - this.logger.info(`Map manager loaded with ${Object.keys(this.maps).length} maps`) } - public async loadMap(map: Map): Promise { - this.maps[map.id] = new LoadedMap(map) + public async loadMap(map: Map): Promise { + const loadedMap = new LoadedMap(map) + this.maps[map.id] = loadedMap this.logger.info(`Map ID ${map.id} loaded`) + return loadedMap } public unloadMap(mapId: UUID): void { + const map = this.maps[mapId] + if (!map) return + delete this.maps[mapId] this.logger.info(`Map ID ${mapId} unloaded`) } @@ -38,15 +42,11 @@ class MapManager { } public getCharacterById(characterId: UUID): MapCharacter | null { - return ( - Object.values(this.maps) - .flatMap((map) => map.getCharactersInMap()) - .find((char) => char.character.id === characterId) ?? null - ) - } - - public removeCharacter(characterId: UUID): void { - Object.values(this.maps).forEach((map) => map.removeCharacter(characterId)) + for (const map of Object.values(this.maps)) { + const character = map.getCharacterById(characterId) + if (character) return character + } + return null } } diff --git a/src/managers/socketManager.ts b/src/managers/socketManager.ts index 3558dd9..9507678 100644 --- a/src/managers/socketManager.ts +++ b/src/managers/socketManager.ts @@ -1,6 +1,7 @@ import fs from 'fs' import { Server as HTTPServer } from 'http' import { pathToFileURL } from 'url' +import { SocketEvent } from '@/application/enums' import Logger, { LoggerType } from '@/application/logger' import Storage from '@/application/storage' import type { TSocket, UUID } from '@/application/types' @@ -21,7 +22,7 @@ class SocketManager { this.io.use(Authentication) // Set up connection handler - this.io.on('connection', this.handleConnection.bind(this)) + this.io.on(SocketEvent.CONNECTION, this.handleConnection.bind(this)) } /** diff --git a/src/migrations/.snapshot-game.json b/src/migrations/.snapshot-game.json index c610141..c9f81f2 100644 --- a/src/migrations/.snapshot-game.json +++ b/src/migrations/.snapshot-game.json @@ -21,6 +21,7 @@ "primary": false, "nullable": false, "length": 255, + "default": "''", "mappedType": "string" }, "width": { @@ -51,7 +52,7 @@ "unsigned": false, "autoincrement": false, "primary": false, - "nullable": true, + "nullable": false, "length": null, "mappedType": "json" }, @@ -233,6 +234,7 @@ "primary": false, "nullable": false, "length": null, + "default": "0", "mappedType": "integer" }, "position_y": { @@ -243,6 +245,7 @@ "primary": false, "nullable": false, "length": null, + "default": "0", "mappedType": "integer" }, "teleport_id": { @@ -314,7 +317,7 @@ "id" ], "referencedTableName": "map_event_tile_teleport", - "deleteRule": "set null", + "deleteRule": "cascade", "updateRule": "cascade" } }, @@ -736,6 +739,26 @@ "length": 255, "mappedType": "string" }, + "width": { + "name": "width", + "type": "int", + "unsigned": false, + "autoincrement": false, + "primary": false, + "nullable": true, + "length": null, + "mappedType": "integer" + }, + "height": { + "name": "height", + "type": "int", + "unsigned": false, + "autoincrement": false, + "primary": false, + "nullable": true, + "length": null, + "mappedType": "integer" + }, "created_at": { "name": "created_at", "type": "datetime", @@ -860,7 +883,7 @@ "unsigned": false, "autoincrement": false, "primary": false, - "nullable": true, + "nullable": false, "length": 255, "mappedType": "string" }, @@ -920,7 +943,6 @@ "id" ], "referencedTableName": "sprite", - "deleteRule": "set null", "updateRule": "cascade" } }, @@ -1093,6 +1115,17 @@ "default": "'MALE'", "mappedType": "string" }, + "color": { + "name": "color", + "type": "varchar(255)", + "unsigned": false, + "autoincrement": false, + "primary": false, + "nullable": false, + "length": 255, + "default": "'#000000'", + "mappedType": "string" + }, "is_selectable": { "name": "is_selectable", "type": "tinyint(1)", @@ -1110,7 +1143,7 @@ "unsigned": false, "autoincrement": false, "primary": false, - "nullable": true, + "nullable": false, "length": 255, "mappedType": "string" }, @@ -1170,7 +1203,6 @@ "id" ], "referencedTableName": "sprite", - "deleteRule": "set null", "updateRule": "cascade" } }, diff --git a/src/migrations/Migration20250215152716.ts b/src/migrations/Migration20250221004940.ts similarity index 92% rename from src/migrations/Migration20250215152716.ts rename to src/migrations/Migration20250221004940.ts index 49e2473..9b33416 100644 --- a/src/migrations/Migration20250215152716.ts +++ b/src/migrations/Migration20250221004940.ts @@ -1,14 +1,14 @@ import { Migration } from '@mikro-orm/migrations'; -export class Migration20250215152716 extends Migration { +export class Migration20250221004940 extends Migration { override async up(): Promise { - this.addSql(`create table \`map\` (\`id\` varchar(255) not null, \`name\` varchar(255) not null, \`width\` int not null default 10, \`height\` int not null default 10, \`tiles\` json null, \`pvp\` tinyint(1) not null default false, \`created_at\` datetime not null, \`updated_at\` datetime not null, primary key (\`id\`)) default character set utf8mb4 engine = InnoDB;`); + this.addSql(`create table \`map\` (\`id\` varchar(255) not null, \`name\` varchar(255) not null default '', \`width\` int not null default 10, \`height\` int not null default 10, \`tiles\` json not null, \`pvp\` tinyint(1) not null default false, \`created_at\` datetime not null, \`updated_at\` datetime not null, primary key (\`id\`)) default character set utf8mb4 engine = InnoDB;`); this.addSql(`create table \`map_effect\` (\`id\` varchar(255) not null, \`map_id\` varchar(255) not null, \`effect\` varchar(255) not null, \`strength\` int not null, primary key (\`id\`)) default character set utf8mb4 engine = InnoDB;`); this.addSql(`alter table \`map_effect\` add index \`map_effect_map_id_index\`(\`map_id\`);`); - this.addSql(`create table \`map_event_tile\` (\`id\` varchar(255) not null, \`map_id\` varchar(255) not null, \`type\` enum('BLOCK', 'TELEPORT', 'NPC', 'ITEM') not null, \`position_x\` int not null, \`position_y\` int not null, \`teleport_id\` varchar(255) null, primary key (\`id\`)) default character set utf8mb4 engine = InnoDB;`); + this.addSql(`create table \`map_event_tile\` (\`id\` varchar(255) not null, \`map_id\` varchar(255) not null, \`type\` enum('BLOCK', 'TELEPORT', 'NPC', 'ITEM') not null, \`position_x\` int not null default 0, \`position_y\` int not null default 0, \`teleport_id\` varchar(255) null, primary key (\`id\`)) default character set utf8mb4 engine = InnoDB;`); this.addSql(`alter table \`map_event_tile\` add index \`map_event_tile_map_id_index\`(\`map_id\`);`); this.addSql(`alter table \`map_event_tile\` add unique \`map_event_tile_teleport_id_unique\`(\`teleport_id\`);`); @@ -22,15 +22,15 @@ export class Migration20250215152716 extends Migration { this.addSql(`alter table \`placed_map_object\` add index \`placed_map_object_map_id_index\`(\`map_id\`);`); this.addSql(`alter table \`placed_map_object\` add index \`placed_map_object_map_object_id_index\`(\`map_object_id\`);`); - this.addSql(`create table \`sprite\` (\`id\` varchar(255) not null, \`name\` varchar(255) not null, \`created_at\` datetime not null, \`updated_at\` datetime not null, primary key (\`id\`)) default character set utf8mb4 engine = InnoDB;`); + this.addSql(`create table \`sprite\` (\`id\` varchar(255) not null, \`name\` varchar(255) not null, \`width\` int null, \`height\` int null, \`created_at\` datetime not null, \`updated_at\` datetime not null, primary key (\`id\`)) default character set utf8mb4 engine = InnoDB;`); - this.addSql(`create table \`item\` (\`id\` varchar(255) not null, \`name\` varchar(255) not null, \`description\` varchar(255) not null default '', \`item_type\` enum('WEAPON', 'HELMET', 'CHEST', 'LEGS', 'BOOTS', 'GLOVES', 'RING', 'NECKLACE') not null, \`stackable\` tinyint(1) not null default false, \`rarity\` enum('COMMON', 'UNCOMMON', 'RARE', 'EPIC', 'LEGENDARY') not null default 'COMMON', \`sprite_id\` varchar(255) null, \`created_at\` datetime not null, \`updated_at\` datetime not null, primary key (\`id\`)) default character set utf8mb4 engine = InnoDB;`); + this.addSql(`create table \`item\` (\`id\` varchar(255) not null, \`name\` varchar(255) not null, \`description\` varchar(255) not null default '', \`item_type\` enum('WEAPON', 'HELMET', 'CHEST', 'LEGS', 'BOOTS', 'GLOVES', 'RING', 'NECKLACE') not null, \`stackable\` tinyint(1) not null default false, \`rarity\` enum('COMMON', 'UNCOMMON', 'RARE', 'EPIC', 'LEGENDARY') not null default 'COMMON', \`sprite_id\` varchar(255) not null, \`created_at\` datetime not null, \`updated_at\` datetime not null, primary key (\`id\`)) default character set utf8mb4 engine = InnoDB;`); this.addSql(`alter table \`item\` add index \`item_sprite_id_index\`(\`sprite_id\`);`); this.addSql(`create table \`character_type\` (\`id\` varchar(255) not null, \`name\` varchar(255) not null, \`gender\` enum('MALE', 'FEMALE') not null, \`race\` enum('HUMAN', 'ELF', 'DWARF', 'ORC', 'GOBLIN') not null, \`is_selectable\` tinyint(1) not null default false, \`sprite_id\` varchar(255) null, \`created_at\` datetime not null, \`updated_at\` datetime not null, primary key (\`id\`)) default character set utf8mb4 engine = InnoDB;`); this.addSql(`alter table \`character_type\` add index \`character_type_sprite_id_index\`(\`sprite_id\`);`); - this.addSql(`create table \`character_hair\` (\`id\` varchar(255) not null, \`name\` varchar(255) not null, \`gender\` varchar(255) not null default 'MALE', \`is_selectable\` tinyint(1) not null default false, \`sprite_id\` varchar(255) null, \`created_at\` datetime not null, \`updated_at\` datetime not null, primary key (\`id\`)) default character set utf8mb4 engine = InnoDB;`); + this.addSql(`create table \`character_hair\` (\`id\` varchar(255) not null, \`name\` varchar(255) not null, \`gender\` varchar(255) not null default 'MALE', \`color\` varchar(255) not null default '#000000', \`is_selectable\` tinyint(1) not null default false, \`sprite_id\` varchar(255) not null, \`created_at\` datetime not null, \`updated_at\` datetime not null, primary key (\`id\`)) default character set utf8mb4 engine = InnoDB;`); this.addSql(`alter table \`character_hair\` add index \`character_hair_sprite_id_index\`(\`sprite_id\`);`); this.addSql(`create table \`sprite_action\` (\`id\` varchar(255) not null, \`sprite_id\` varchar(255) not null, \`action\` varchar(255) not null, \`sprites\` json null, \`origin_x\` numeric(5,2) not null default 0, \`origin_y\` numeric(5,2) not null default 0, \`frame_width\` int not null default 0, \`frame_height\` int not null default 0, \`frame_rate\` int not null default 0, primary key (\`id\`)) default character set utf8mb4 engine = InnoDB;`); @@ -70,7 +70,7 @@ export class Migration20250215152716 extends Migration { this.addSql(`alter table \`map_effect\` add constraint \`map_effect_map_id_foreign\` foreign key (\`map_id\`) references \`map\` (\`id\`) on update cascade on delete cascade;`); this.addSql(`alter table \`map_event_tile\` add constraint \`map_event_tile_map_id_foreign\` foreign key (\`map_id\`) references \`map\` (\`id\`) on update cascade on delete cascade;`); - this.addSql(`alter table \`map_event_tile\` add constraint \`map_event_tile_teleport_id_foreign\` foreign key (\`teleport_id\`) references \`map_event_tile_teleport\` (\`id\`) on update cascade on delete set null;`); + this.addSql(`alter table \`map_event_tile\` add constraint \`map_event_tile_teleport_id_foreign\` foreign key (\`teleport_id\`) references \`map_event_tile_teleport\` (\`id\`) on update cascade on delete cascade;`); this.addSql(`alter table \`map_event_tile_teleport\` add constraint \`map_event_tile_teleport_map_event_tile_id_foreign\` foreign key (\`map_event_tile_id\`) references \`map_event_tile\` (\`id\`) on update cascade on delete cascade;`); this.addSql(`alter table \`map_event_tile_teleport\` add constraint \`map_event_tile_teleport_to_map_id_foreign\` foreign key (\`to_map_id\`) references \`map\` (\`id\`) on update cascade on delete cascade;`); @@ -78,11 +78,11 @@ export class Migration20250215152716 extends Migration { this.addSql(`alter table \`placed_map_object\` add constraint \`placed_map_object_map_id_foreign\` foreign key (\`map_id\`) references \`map\` (\`id\`) on update cascade on delete cascade;`); this.addSql(`alter table \`placed_map_object\` add constraint \`placed_map_object_map_object_id_foreign\` foreign key (\`map_object_id\`) references \`map_object\` (\`id\`) on update cascade on delete cascade;`); - this.addSql(`alter table \`item\` add constraint \`item_sprite_id_foreign\` foreign key (\`sprite_id\`) references \`sprite\` (\`id\`) on update cascade on delete set null;`); + this.addSql(`alter table \`item\` add constraint \`item_sprite_id_foreign\` foreign key (\`sprite_id\`) references \`sprite\` (\`id\`) on update cascade;`); this.addSql(`alter table \`character_type\` add constraint \`character_type_sprite_id_foreign\` foreign key (\`sprite_id\`) references \`sprite\` (\`id\`) on update cascade on delete set null;`); - this.addSql(`alter table \`character_hair\` add constraint \`character_hair_sprite_id_foreign\` foreign key (\`sprite_id\`) references \`sprite\` (\`id\`) on update cascade on delete set null;`); + this.addSql(`alter table \`character_hair\` add constraint \`character_hair_sprite_id_foreign\` foreign key (\`sprite_id\`) references \`sprite\` (\`id\`) on update cascade;`); this.addSql(`alter table \`sprite_action\` add constraint \`sprite_action_sprite_id_foreign\` foreign key (\`sprite_id\`) references \`sprite\` (\`id\`) on update cascade on delete cascade;`); diff --git a/src/mikro-orm.config.ts b/src/mikro-orm.config.ts index 1274891..5f21b31 100644 --- a/src/mikro-orm.config.ts +++ b/src/mikro-orm.config.ts @@ -1,13 +1,13 @@ import serverConfig from '@/application/config' +import { Migrator } from '@mikro-orm/migrations' import { defineConfig, MariaDbDriver } from '@mikro-orm/mariadb' // import { defineConfig, MySqlDriver } from '@mikro-orm/mysql' -import { Migrator } from '@mikro-orm/migrations' import { TsMorphMetadataProvider } from '@mikro-orm/reflection' export default defineConfig({ extensions: [Migrator], metadataProvider: TsMorphMetadataProvider, - entities: ['./dist/entities/*.js'], + entities: ['./src/entities/*.ts'], entitiesTs: ['./src/entities/*.ts'], driver: MariaDbDriver, host: serverConfig.DB_HOST, diff --git a/src/models/loadedMap.ts b/src/models/loadedMap.ts index f710eec..a2c0c7a 100644 --- a/src/models/loadedMap.ts +++ b/src/models/loadedMap.ts @@ -16,17 +16,23 @@ class LoadedMap { return this.map } - public addCharacter(character: Character) { + public addCharacter(character: Character): MapCharacter { + const existingCharacter = this.getCharacterById(character.id) + if (existingCharacter) { + return existingCharacter + } + const mapCharacter = new MapCharacter(character) this.characters.push(mapCharacter) + return mapCharacter } - public async removeCharacter(id: UUID) { + public async removeCharacter(id: UUID): Promise { const mapCharacter = this.getCharacterById(id) - if (mapCharacter) { - await mapCharacter.savePosition() - this.characters = this.characters.filter((c) => c.character.id !== id) - } + if (!mapCharacter) return + + await mapCharacter.savePosition() + this.characters = this.characters.filter((c) => c.character.id !== id) } public getCharacterById(id: UUID): MapCharacter | undefined { @@ -38,12 +44,11 @@ class LoadedMap { } public async getGrid(): Promise { - let grid: number[][] = Array.from({ length: this.map.height }, () => Array.from({ length: this.map.width }, () => 0)) + const grid: number[][] = Array.from({ length: this.map.height }, () => Array.from({ length: this.map.width }, () => 0)) const mapEventTileRepository = new MapEventTileRepository() const eventTiles = await mapEventTileRepository.getAll(this.map.id) - // Set the grid values based on the event tiles, these are strings eventTiles.forEach((eventTile) => { if (eventTile.type === 'BLOCK') { grid[eventTile.positionY]![eventTile.positionX] = 1 diff --git a/src/models/mapCharacter.ts b/src/models/mapCharacter.ts index 5855a47..560fa89 100644 --- a/src/models/mapCharacter.ts +++ b/src/models/mapCharacter.ts @@ -38,14 +38,16 @@ class MapCharacter { await this.savePosition() // Leave map and remove from manager - if (this.character.map) { - socket.leave(this.character.map.id) - MapManager.removeCharacter(this.character.id) + socket.leave(this.character.map.id) - // Notify map players - io.in(this.character.map.id).emit(SocketEvent.MAP_CHARACTER_LEAVE, this.character.id) + const map = MapManager.getMapById(this.character.map.id) + if (map) { + await map.removeCharacter(this.character.id) } + // Notify map players + io.in(this.character.map.id).emit(SocketEvent.MAP_CHARACTER_LEAVE, this.character.id) + // Notify all players io.emit(SocketEvent.CHARACTER_DISCONNECT, this.character.id) } catch (error) { diff --git a/src/services/characterMoveService.ts b/src/services/characterMoveService.ts index 92756f9..9b3a56a 100644 --- a/src/services/characterMoveService.ts +++ b/src/services/characterMoveService.ts @@ -1,75 +1,17 @@ import { BaseService } from '@/application/base/baseService' import config from '@/application/config' +import { SocketEvent } from '@/application/enums' +import type { MapEventTileWithTeleport, UUID } from '@/application/types' import { Character } from '@/entities/character' import MapManager from '@/managers/mapManager' +import SocketManager from '@/managers/socketManager' +import MapEventTileRepository from '@/repositories/mapEventTileRepository' +import TeleportService from '@/services/characterTeleportService' +import type { Server } from 'socket.io' type Position = { positionX: number; positionY: number } export type Node = Position & { parent?: Node; g: number; h: number; f: number } -class PriorityQueue { - private items: T[] = [] - constructor(private compare: (a: T, b: T) => number) {} - - enqueue(item: T): void { - this.items.push(item) - this.siftUp(this.items.length - 1) - } - - dequeue(): T | undefined { - if (this.items.length === 0) return undefined - - const result = this.items[0] - const last = this.items.pop() - - if (this.items.length > 0 && last !== undefined) { - this.items[0] = last - this.siftDown(0) - } - - return result - } - - get length(): number { - return this.items.length - } - - private siftUp(index: number): void { - while (index > 0) { - const parentIndex = Math.floor((index - 1) / 2) - if (this.compare(this.items[index]!, this.items[parentIndex]!) < 0) { - ;[this.items[index], this.items[parentIndex]] = [this.items[parentIndex]!, this.items[index]!] - index = parentIndex - } else { - break - } - } - } - - private siftDown(index: number): void { - const length = this.items.length - while (true) { - let minIndex = index - const leftChild = 2 * index + 1 - const rightChild = 2 * index + 2 - - if (leftChild < length && this.compare(this.items[leftChild]!, this.items[minIndex]!) < 0) { - minIndex = leftChild - } - if (rightChild < length && this.compare(this.items[rightChild]!, this.items[minIndex]!) < 0) { - minIndex = rightChild - } - - if (minIndex === index) break - ;[this.items[index], this.items[minIndex]] = [this.items[minIndex]!, this.items[index]!] - index = minIndex - } - } - - clear(): void { - this.items = [] - } -} - class CharacterMoveService extends BaseService { // Rotation lookup table for better performance private readonly ROTATION_MAP = { @@ -174,12 +116,14 @@ class CharacterMoveService extends BaseService { private findPath(start: Position, end: Position, grid: number[][]): Node[] { const openList = new PriorityQueue((a, b) => a.f - b.f) const closedSet = new Set() - const MAX_ITERATIONS = 1000 // Prevent infinite loops for impossible paths + const MAX_ITERATIONS = 1000 let iterations = 0 const getKey = (p: Position) => `${p.positionX},${p.positionY}` + const gScoreMap = new Map() - openList.enqueue({ ...start, g: 0, h: 0, f: 0 }) + openList.enqueue({ ...start, g: 0, h: this.getDistance(start, end), f: this.getDistance(start, end) }) + gScoreMap.set(getKey(start), 0) try { while (openList.length > 0 && iterations < MAX_ITERATIONS) { @@ -191,28 +135,34 @@ class CharacterMoveService extends BaseService { return this.reconstructPath(current) } - closedSet.add(getKey(current)) + const currentKey = getKey(current) + closedSet.add(currentKey) const neighbors = this.getValidNeighbors(current, grid, end) for (const neighbor of neighbors) { - if (closedSet.has(getKey(neighbor))) continue + const neighborKey = getKey(neighbor) + if (closedSet.has(neighborKey)) continue - const g = current.g + this.getDistance(current, neighbor) - const h = this.getDistance(neighbor, end) - const f = g + h + const tentativeGScore = current.g + this.getDistance(current, neighbor) - const node: Node = { ...neighbor, g, h, f, parent: current } - openList.enqueue(node) + if (!gScoreMap.has(neighborKey) || tentativeGScore < gScoreMap.get(neighborKey)!) { + gScoreMap.set(neighborKey, tentativeGScore) + const h = this.getDistance(neighbor, end) + const f = tentativeGScore + h + + const node: Node = { ...neighbor, g: tentativeGScore, h, f, parent: current } + openList.enqueue(node) + } } } this.logger.warn(`Path not found after ${iterations} iterations`) - return [] // No path found + return [] } finally { - // Clean up resources - while (openList.length > 0) openList.dequeue() + openList.clear() closedSet.clear() + gScoreMap.clear() } } @@ -226,13 +176,32 @@ class CharacterMoveService extends BaseService { } private isValidPosition(pos: Position, grid: number[][], end: Position): boolean { - return ( - pos.positionX >= 0 && - pos.positionY >= 0 && - pos.positionX < grid[0]!.length && - pos.positionY < grid.length && - (grid[pos.positionY]![pos.positionX] === 0 || (pos.positionX === end.positionX && pos.positionY === end.positionY)) - ) + if (pos.positionX < 0 || pos.positionY < 0 || pos.positionX >= grid[0]!.length || pos.positionY >= grid.length) { + return false + } + + return grid[pos.positionY]![pos.positionX] === 0 || (pos.positionX === end.positionX && pos.positionY === end.positionY) + } + + isValidStep(current: { positionX: number; positionY: number }, next: { positionX: number; positionY: number }): boolean { + return Math.abs(next.positionX - current.positionX) <= 1 && Math.abs(next.positionY - current.positionY) <= 1 + } + + async checkMapEvents(map_id: UUID, nextTile: { positionX: number; positionY: number }) { + const mapEventTileRepository = new MapEventTileRepository() + return mapEventTileRepository.getEventTileByMapIdAndPosition(map_id, nextTile.positionX, nextTile.positionY) + } + + async handleTeleportMapEventTile(character_id: UUID, mapEventTile: MapEventTileWithTeleport): Promise { + const teleport = mapEventTile.getTeleport() + if (teleport) { + await TeleportService.teleportCharacter(character_id, { + targetMapId: teleport.getToMap().getId(), + targetX: teleport.getToPositionX(), + targetY: teleport.getToPositionY(), + rotation: teleport.getToRotation() + }) + } } private getDistance(a: Position, b: Position): number { @@ -252,6 +221,11 @@ class CharacterMoveService extends BaseService { return path } + public broadcastMovement(character: Character, isMoving: boolean = false): void { + const io: Server = SocketManager.getIO() + io.in(character.map.id).emit(SocketEvent.MAP_CHARACTER_MOVE, [character.id, character.getPositionX(), character.getPositionY(), character.getRotation(), isMoving]) + } + public validateMovementDistance( currentX: number, currentY: number, @@ -263,7 +237,7 @@ class CharacterMoveService extends BaseService { return { isValid: true, maxAllowedDistance: 0, actualDistance: 0 } } - const maxAllowedDistance = Math.ceil((stepDelay / 1000) * 2) + const maxAllowedDistance = Math.ceil((stepDelay / 1000) * (config.ALLOW_DIAGONAL_MOVEMENT ? 2.5 : 2)) const actualDistance = Math.sqrt(Math.pow(currentX - lastKnownPosition.x, 2) + Math.pow(currentY - lastKnownPosition.y, 2)) return { @@ -272,18 +246,70 @@ class CharacterMoveService extends BaseService { actualDistance } } - - public validateRequestDistance(currentX: number, currentY: number, targetX: number, targetY: number, maxRequestDistance: number): { isValid: boolean; distance: number } { - const requestDistance = Math.sqrt(Math.pow(targetX - currentX, 2) + Math.pow(targetY - currentY, 2)) - return { - isValid: requestDistance <= maxRequestDistance, - distance: requestDistance - } - } - - public isValidStep(current: { positionX: number; positionY: number }, next: { positionX: number; positionY: number }): boolean { - return Math.abs(next.positionX - current.positionX) <= 1 && Math.abs(next.positionY - current.positionY) <= 1 - } } export default new CharacterMoveService() + +class PriorityQueue { + private items: T[] = [] + constructor(private compare: (a: T, b: T) => number) {} + + enqueue(item: T): void { + this.items.push(item) + this.siftUp(this.items.length - 1) + } + + dequeue(): T | undefined { + if (this.items.length === 0) return undefined + + const result = this.items[0] + const last = this.items.pop() + + if (this.items.length > 0 && last !== undefined) { + this.items[0] = last + this.siftDown(0) + } + + return result + } + + get length(): number { + return this.items.length + } + + private siftUp(index: number): void { + while (index > 0) { + const parentIndex = Math.floor((index - 1) / 2) + if (this.compare(this.items[index]!, this.items[parentIndex]!) < 0) { + ;[this.items[index], this.items[parentIndex]] = [this.items[parentIndex]!, this.items[index]!] + index = parentIndex + } else { + break + } + } + } + + private siftDown(index: number): void { + const length = this.items.length + while (true) { + let minIndex = index + const leftChild = 2 * index + 1 + const rightChild = 2 * index + 2 + + if (leftChild < length && this.compare(this.items[leftChild]!, this.items[minIndex]!) < 0) { + minIndex = leftChild + } + if (rightChild < length && this.compare(this.items[rightChild]!, this.items[minIndex]!) < 0) { + minIndex = rightChild + } + + if (minIndex === index) break + ;[this.items[index], this.items[minIndex]] = [this.items[minIndex]!, this.items[index]!] + index = minIndex + } + } + + clear(): void { + this.items = [] + } +} diff --git a/src/services/characterTeleportService.ts b/src/services/characterTeleportService.ts index f6f1b00..646a6e7 100644 --- a/src/services/characterTeleportService.ts +++ b/src/services/characterTeleportService.ts @@ -6,6 +6,7 @@ import MapManager from '@/managers/mapManager' import SocketManager from '@/managers/socketManager' import MapCharacter from '@/models/mapCharacter' import MapRepository from '@/repositories/mapRepository' +import CharacterMoveService from '@/services/characterMoveService' interface TeleportOptions { targetMapId: UUID @@ -20,77 +21,25 @@ class CharacterTeleportService { private readonly logger = Logger.type(LoggerType.GAME) public async teleportCharacter(characterId: UUID, options: TeleportOptions): Promise { - const mapRepository = new MapRepository() - const socket = SocketManager.getSocketByCharacterId(characterId) - const targetMap = MapManager.getMapById(options.targetMapId) - - if (!socket || !targetMap) { - this.logger.error(`Teleport failed - Missing socket or target map for character ${characterId}`) - return false - } - - if (options.isInitialJoin && !options.character) { - this.logger.error('Initial join requires character data') - return false - } - - const existingCharacter = !options.isInitialJoin && MapManager.getCharacterById(characterId) - const mapCharacter = options.isInitialJoin - ? new MapCharacter(options.character) - : existingCharacter || - (() => { - this.logger.error(`Teleport failed - Character ${characterId} not found in MapManager`) - return null - })() - - if (!mapCharacter) return false - try { - const currentMapId = mapCharacter.character.map?.id + const { socket, targetMap, mapCharacter } = await this.validateTeleportRequest(characterId, options) + if (!socket || !targetMap || !mapCharacter) return false + + const currentMapId = mapCharacter.character.map.id + const currentMap = MapManager.getMapById(currentMapId) const io = SocketManager.getIO() // Update character position and map - await mapCharacter - .getCharacter() - .setPositionX(options.targetX) - .setPositionY(options.targetY) - .setRotation(options.rotation ?? 0) - .setMap(targetMap.getMap()) - .save() + await this.updateCharacterPosition(mapCharacter, options, targetMap) - // If the current map is the target map and we are not joining, send move event + // Handle same map teleport if (currentMapId === options.targetMapId && !options.isInitialJoin) { - // If the current map is the target map, send move event - io.in(currentMapId).emit(SocketEvent.MAP_CHARACTER_MOVE, { - characterId: mapCharacter.character.id, - positionX: options.targetX, - positionY: options.targetY, - rotation: options.rotation ?? 0, - isMoving: false - }) + CharacterMoveService.broadcastMovement(mapCharacter.character, false) return true } - // Handle current map cleanup - if (currentMapId) { - socket.leave(currentMapId) - MapManager.removeCharacter(characterId) - io.in(currentMapId).emit(SocketEvent.MAP_CHARACTER_LEAVE, characterId) - } - - // Join new map - socket.join(options.targetMapId) - targetMap.addCharacter(mapCharacter.getCharacter()) - - const map = await mapRepository.getById(options.targetMapId) - await mapRepository.getEntityManager().populate(map, mapRepository.POPULATE_TELEPORT as any) - - // Notify clients - io.in(options.targetMapId).emit(SocketEvent.MAP_CHARACTER_JOIN, mapCharacter) - socket.emit(SocketEvent.MAP_CHARACTER_TELEPORT, { - mapId: options.targetMapId, - characters: targetMap.getCharactersInMap() - }) + // Handle map transition + await this.handleMapTransition(socket, io, currentMapId, currentMap, options.targetMapId, targetMap, characterId, mapCharacter) return true } catch (error) { @@ -98,6 +47,64 @@ class CharacterTeleportService { return false } } + + private async validateTeleportRequest(characterId: UUID, options: TeleportOptions) { + const socket = SocketManager.getSocketByCharacterId(characterId) + const targetMap = MapManager.getMapById(options.targetMapId) + + if (!socket || !targetMap) { + this.logger.error(`Teleport failed - Missing socket or target map for character ${characterId}`) + return { socket: null, targetMap: null, mapCharacter: null } + } + + if (options.isInitialJoin && !options.character) { + this.logger.error('Initial join requires character data') + return { socket, targetMap, mapCharacter: null } + } + + const existingCharacter = !options.isInitialJoin && MapManager.getCharacterById(characterId) + const mapCharacter = options.isInitialJoin ? new MapCharacter(options.character!) : existingCharacter || null + + if (!mapCharacter) { + this.logger.error(`Teleport failed - Character ${characterId} not found in MapManager`) + } + + return { socket, targetMap, mapCharacter } + } + + private async updateCharacterPosition(mapCharacter: MapCharacter, options: TeleportOptions, targetMap: any) { + await mapCharacter + .getCharacter() + .setPositionX(options.targetX) + .setPositionY(options.targetY) + .setRotation(options.rotation ?? 0) + .setMap(targetMap.getMap()) + .save() + } + + private async handleMapTransition(socket: any, io: any, currentMapId: UUID | undefined, currentMap: any, targetMapId: UUID, targetMap: any, characterId: UUID, mapCharacter: MapCharacter) { + // Clean up current map + if (currentMapId && currentMap) { + socket.leave(currentMapId) + await currentMap.removeCharacter(characterId) + io.in(currentMapId).emit(SocketEvent.MAP_CHARACTER_LEAVE, characterId) + } + + // Join new map + socket.join(targetMapId) + targetMap.addCharacter(mapCharacter.getCharacter()) + + const mapRepository = new MapRepository() + const map = await mapRepository.getById(targetMapId) + await mapRepository.getEntityManager().populate(map!, mapRepository.POPULATE_TELEPORT as any) + + // Notify clients + io.in(targetMapId).emit(SocketEvent.MAP_CHARACTER_JOIN, mapCharacter) + socket.emit(SocketEvent.MAP_CHARACTER_TELEPORT, { + mapId: targetMapId, + characters: targetMap.getCharactersInMap() + }) + } } export default new CharacterTeleportService()