diff --git a/.gitignore b/.gitignore index ccb5166..b5e703e 100644 --- a/.gitignore +++ b/.gitignore @@ -1,2 +1,3 @@ /target -.vscode \ No newline at end of file +.vscode +.env \ No newline at end of file diff --git a/Cargo.lock b/Cargo.lock index 7570bb5..805e95d 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -29,6 +29,15 @@ dependencies = [ "zerocopy", ] +[[package]] +name = "aho-corasick" +version = "1.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e60d3430d3a69478ad0993f19238d2df97c507009a52b3c10addcd7f6bcb916" +dependencies = [ + "memchr", +] + [[package]] name = "allocator-api2" version = "0.2.18" @@ -77,7 +86,7 @@ dependencies = [ "hyper", "hyper-util", "itoa", - "matchit", + "matchit 0.7.3", "memchr", "mime", "percent-encoding", @@ -149,10 +158,16 @@ version = "0.1.0" dependencies = [ "axum", "dotenv", + "futures-util", "rand", "serde", + "serde_json", + "socketioxide", "sqlx", "tokio", + "tower-http", + "tracing", + "tracing-subscriber", ] [[package]] @@ -184,6 +199,9 @@ name = "bytes" version = "1.7.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8318a53db07bb3f8dca91a600466bdb3f2eaadeedfdbcf02e1accbad9271ba50" +dependencies = [ + "serde", +] [[package]] name = "cc" @@ -264,6 +282,12 @@ dependencies = [ "typenum", ] +[[package]] +name = "data-encoding" +version = "2.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e8566979429cf69b49a5c740c60791108e86440e8be149bbea4fe54d2c32d6e2" + [[package]] name = "der" version = "0.7.9" @@ -308,6 +332,33 @@ dependencies = [ "serde", ] +[[package]] +name = "engineioxide" +version = "0.14.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c9af9d31fdcf7ae420195c6df199b58372c74f4a1966f7107ebfbebbfceafa16" +dependencies = [ + "base64", + "bytes", + "futures-core", + "futures-util", + "http", + "http-body", + "http-body-util", + "hyper", + "hyper-util", + "pin-project-lite", + "rand", + "serde", + "serde_json", + "smallvec", + "thiserror", + "tokio", + "tokio-tungstenite", + "tower", + "tracing", +] + [[package]] name = "equivalent" version = "1.0.1" @@ -422,6 +473,17 @@ version = "0.3.30" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a44623e20b9681a318efdd71c299b6b222ed6f231972bfe2f224ebad6311f0c1" +[[package]] +name = "futures-macro" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87750cf4b7a4c0625b1529e4c543c2182106e4dedc60a2a6455e00d212c489ac" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + [[package]] name = "futures-sink" version = "0.3.30" @@ -442,6 +504,7 @@ checksum = "3d6401deb83407ab3da39eba7e33987a73c3df0c82b4bb5813ee871c19c41d48" dependencies = [ "futures-core", "futures-io", + "futures-macro", "futures-sink", "futures-task", "memchr", @@ -450,6 +513,19 @@ dependencies = [ "slab", ] +[[package]] +name = "generator" +version = "0.7.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5cc16584ff22b460a382b7feec54b23d2908d858152e5739a120b949293bd74e" +dependencies = [ + "cc", + "libc", + "log", + "rustversion", + "windows", +] + [[package]] name = "generic-array" version = "0.14.7" @@ -701,12 +777,42 @@ version = "0.4.22" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a7a70ba024b9dc04c27ea2f0c0548feb474ec5c54bba33a7f72f873a39d07b24" +[[package]] +name = "loom" +version = "0.5.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ff50ecb28bb86013e935fb6683ab1f6d3a20016f123c76fd4c27470076ac30f5" +dependencies = [ + "cfg-if", + "generator", + "scoped-tls", + "serde", + "serde_json", + "tracing", + "tracing-subscriber", +] + +[[package]] +name = "matchers" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8263075bb86c5a1b1427b5ae862e8889656f126e9f77c484496e8b47cf5c5558" +dependencies = [ + "regex-automata 0.1.10", +] + [[package]] name = "matchit" version = "0.7.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0e7465ac9959cc2b1404e8e2367b43684a6d13790fe23056cc8c6c5a6b7bcb94" +[[package]] +name = "matchit" +version = "0.8.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "47e1ffaa40ddd1f3ed91f717a33c8c0ee23fff369e3aa8772b9605cc1d22f4c3" + [[package]] name = "md-5" version = "0.10.6" @@ -766,6 +872,16 @@ dependencies = [ "minimal-lexical", ] +[[package]] +name = "nu-ansi-term" +version = "0.46.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "77a8165726e8236064dbb45459242600304b42a5ea24ee2948e18e023bf7ba84" +dependencies = [ + "overload", + "winapi", +] + [[package]] name = "num-bigint-dig" version = "0.8.4" @@ -828,6 +944,12 @@ version = "1.19.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3fdb12b2476b595f9358c5161aa467c2438859caa136dec86c26fdd2efe17b92" +[[package]] +name = "overload" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b15813163c1d831bf4a13c3610c05c0d03b39feb07f7e09fa234dac9b15aaf39" + [[package]] name = "parking" version = "2.2.1" @@ -1003,6 +1125,50 @@ dependencies = [ "bitflags", ] +[[package]] +name = "regex" +version = "1.10.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4219d74c6b67a3654a9fbebc4b419e22126d13d2f3c4a07ee0cb61ff79a79619" +dependencies = [ + "aho-corasick", + "memchr", + "regex-automata 0.4.7", + "regex-syntax 0.8.4", +] + +[[package]] +name = "regex-automata" +version = "0.1.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c230d73fb8d8c1b9c0b3135c5142a8acee3a0558fb8db5cf1cb65f8d7862132" +dependencies = [ + "regex-syntax 0.6.29", +] + +[[package]] +name = "regex-automata" +version = "0.4.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38caf58cc5ef2fed281f89292ef23f6365465ed9a41b7a7754eb4e26496c92df" +dependencies = [ + "aho-corasick", + "memchr", + "regex-syntax 0.8.4", +] + +[[package]] +name = "regex-syntax" +version = "0.6.29" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f162c6dd7b008981e4d40210aca20b4bd0f9b60ca9271061b07f78537722f2e1" + +[[package]] +name = "regex-syntax" +version = "0.8.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7a66a03ae7c801facd77a29370b4faec201768915ac14a721ba36f20bc9c209b" + [[package]] name = "rsa" version = "0.9.6" @@ -1054,6 +1220,12 @@ version = "1.0.18" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f3cb5ba0dc43242ce17de99c180e96db90b235b8a9fdc9543c96d2209116bd9f" +[[package]] +name = "scoped-tls" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e1cf6437eb19a8f4a6cc0f7dca544973b0b78843adbfeb3683d1a94a0024a294" + [[package]] name = "scopeguard" version = "1.2.0" @@ -1136,6 +1308,15 @@ dependencies = [ "digest", ] +[[package]] +name = "sharded-slab" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f40ca3c46823713e0d4209592e8d6e826aa57e928f09752619fc696c499637f6" +dependencies = [ + "lazy_static", +] + [[package]] name = "shlex" version = "1.3.0" @@ -1189,6 +1370,31 @@ dependencies = [ "windows-sys 0.52.0", ] +[[package]] +name = "socketioxide" +version = "0.14.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eae037b680e678f04b270f9740cbe6ddd71cf77a2e89f65092a0fcd1639af561" +dependencies = [ + "bytes", + "engineioxide", + "futures-core", + "futures-util", + "http", + "http-body", + "hyper", + "itoa", + "matchit 0.8.4", + "pin-project-lite", + "serde", + "serde_json", + "state", + "thiserror", + "tokio", + "tower", + "tracing", +] + [[package]] name = "spin" version = "0.9.8" @@ -1264,6 +1470,8 @@ dependencies = [ "smallvec", "sqlformat", "thiserror", + "tokio", + "tokio-stream", "tracing", "url", ] @@ -1303,6 +1511,7 @@ dependencies = [ "sqlx-sqlite", "syn", "tempfile", + "tokio", "url", ] @@ -1409,6 +1618,15 @@ dependencies = [ "url", ] +[[package]] +name = "state" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2b8c4a4445d81357df8b1a650d0d0d6fbbbfe99d064aa5e02f3e4022061476d8" +dependencies = [ + "loom", +] + [[package]] name = "stringprep" version = "0.1.5" @@ -1482,6 +1700,16 @@ dependencies = [ "syn", ] +[[package]] +name = "thread_local" +version = "1.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b9ef9bad013ada3808854ceac7b46812a6465ba368859a37e2100283d2d719c" +dependencies = [ + "cfg-if", + "once_cell", +] + [[package]] name = "tinyvec" version = "1.8.0" @@ -1526,6 +1754,29 @@ dependencies = [ "syn", ] +[[package]] +name = "tokio-stream" +version = "0.1.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4f4e6ce100d0eb49a2734f8c0812bcd324cf357d21810932c5df6b96ef2b86f1" +dependencies = [ + "futures-core", + "pin-project-lite", + "tokio", +] + +[[package]] +name = "tokio-tungstenite" +version = "0.23.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c6989540ced10490aaf14e6bad2e3d33728a2813310a0c71d1574304c49631cd" +dependencies = [ + "futures-util", + "log", + "tokio", + "tungstenite", +] + [[package]] name = "tower" version = "0.4.13" @@ -1542,6 +1793,22 @@ dependencies = [ "tracing", ] +[[package]] +name = "tower-http" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e9cd434a998747dd2c4276bc96ee2e0c7a2eadf3cae88e52be55a05fa9053f5" +dependencies = [ + "bitflags", + "bytes", + "http", + "http-body", + "http-body-util", + "pin-project-lite", + "tower-layer", + "tower-service", +] + [[package]] name = "tower-layer" version = "0.3.3" @@ -1584,6 +1851,54 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c06d3da6113f116aaee68e4d601191614c9053067f9ab7f6edbcb161237daa54" dependencies = [ "once_cell", + "valuable", +] + +[[package]] +name = "tracing-log" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ee855f1f400bd0e5c02d150ae5de3840039a3f54b025156404e34c23c03f47c3" +dependencies = [ + "log", + "once_cell", + "tracing-core", +] + +[[package]] +name = "tracing-subscriber" +version = "0.3.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ad0f048c97dbd9faa9b7df56362b8ebcaa52adb06b498c050d2f4e32f90a7a8b" +dependencies = [ + "matchers", + "nu-ansi-term", + "once_cell", + "regex", + "sharded-slab", + "smallvec", + "thread_local", + "tracing", + "tracing-core", + "tracing-log", +] + +[[package]] +name = "tungstenite" +version = "0.23.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6e2e2ce1e47ed2994fd43b04c8f618008d4cabdd5ee34027cf14f9d918edd9c8" +dependencies = [ + "byteorder", + "bytes", + "data-encoding", + "http", + "httparse", + "log", + "rand", + "sha1", + "thiserror", + "utf-8", ] [[package]] @@ -1636,6 +1951,18 @@ dependencies = [ "percent-encoding", ] +[[package]] +name = "utf-8" +version = "0.7.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09cc8ee72d2a9becf2f2febe0205bbed8fc6615b7cb429ad062dc7b7ddd036a9" + +[[package]] +name = "valuable" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "830b7e5d4d90034032940e4ace0d9a9a057e7a45cd94e6c007832e39edb82f6d" + [[package]] name = "vcpkg" version = "0.2.15" @@ -1670,6 +1997,37 @@ dependencies = [ "wasite", ] +[[package]] +name = "winapi" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419" +dependencies = [ + "winapi-i686-pc-windows-gnu", + "winapi-x86_64-pc-windows-gnu", +] + +[[package]] +name = "winapi-i686-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" + +[[package]] +name = "winapi-x86_64-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" + +[[package]] +name = "windows" +version = "0.48.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e686886bc078bc1b0b600cac0147aadb815089b6e4da64016cbd754b6342700f" +dependencies = [ + "windows-targets 0.48.5", +] + [[package]] name = "windows-sys" version = "0.48.0" diff --git a/Cargo.toml b/Cargo.toml index 3d36bf8..ec78f89 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -6,7 +6,13 @@ edition = "2021" [dependencies] axum = "0.7.5" dotenv = "0.15.0" +futures-util = "0.3.30" rand = "0.8.5" serde = { version = "1.0.210", features = ["derive"] } -sqlx = "0.8.2" +serde_json = "1.0.128" +socketioxide = { version = "0.14.1", features = ["state", "tracing"] } +sqlx = { version = "0.8.2", features = ["macros", "postgres", "runtime-tokio"] } tokio = { version = "1.40.0", features = ["full"] } +tower-http = { version = "0.5.2", features = ["cors"] } +tracing = "0.1.40" +tracing-subscriber = "0.3.18" diff --git a/app/package-lock.json b/app/package-lock.json index 04f7dee..a64720e 100644 --- a/app/package-lock.json +++ b/app/package-lock.json @@ -8,7 +8,8 @@ "name": "app", "version": "0.0.1", "dependencies": { - "lucide-svelte": "^0.441.0" + "lucide-svelte": "^0.441.0", + "socket.io-client": "^4.7.5" }, "devDependencies": { "@sveltejs/adapter-auto": "^3.0.0", @@ -977,6 +978,12 @@ "win32" ] }, + "node_modules/@socket.io/component-emitter": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@socket.io/component-emitter/-/component-emitter-3.1.2.tgz", + "integrity": "sha512-9BCxFwvbGg/RsZK9tjXd8s4UcwR0MWeFQ1XEKIQVVvAGJyINdrqKMcTRyLoK8Rse1GjzLV9cwjWV1olXRWEXVA==", + "license": "MIT" + }, "node_modules/@sveltejs/adapter-auto": { "version": "3.2.4", "resolved": "https://registry.npmjs.org/@sveltejs/adapter-auto/-/adapter-auto-3.2.4.tgz", @@ -1804,7 +1811,6 @@ "version": "4.3.7", "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.7.tgz", "integrity": "sha512-Er2nc/H7RrMXZBFCEim6TCmMk02Z8vLC2Rbi1KEBggpo0fS6l0S1nnapwmIi3yW/+GOJap1Krg4w0Hg80oCqgQ==", - "dev": true, "license": "MIT", "dependencies": { "ms": "^2.1.3" @@ -1886,6 +1892,28 @@ "dev": true, "license": "MIT" }, + "node_modules/engine.io-client": { + "version": "6.5.4", + "resolved": "https://registry.npmjs.org/engine.io-client/-/engine.io-client-6.5.4.tgz", + "integrity": "sha512-GeZeeRjpD2qf49cZQ0Wvh/8NJNfeXkXXcoGh+F77oEAgo9gUHwT1fCRxSNU+YEEaysOJTnsFHmM5oAcPy4ntvQ==", + "license": "MIT", + "dependencies": { + "@socket.io/component-emitter": "~3.1.0", + "debug": "~4.3.1", + "engine.io-parser": "~5.2.1", + "ws": "~8.17.1", + "xmlhttprequest-ssl": "~2.0.0" + } + }, + "node_modules/engine.io-parser": { + "version": "5.2.3", + "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-5.2.3.tgz", + "integrity": "sha512-HqD3yTBfnBxIrbnM1DoD6Pcq8NECnh8d4As1Qgh0z5Gg3jRRIqijury0CL3ghu/edArpUYiYqQiDUQBIs4np3Q==", + "license": "MIT", + "engines": { + "node": ">=10.0.0" + } + }, "node_modules/esbuild": { "version": "0.21.5", "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.21.5.tgz", @@ -2904,7 +2932,6 @@ "version": "2.1.3", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", - "dev": true, "license": "MIT" }, "node_modules/mz": { @@ -3623,6 +3650,34 @@ "node": ">= 10" } }, + "node_modules/socket.io-client": { + "version": "4.7.5", + "resolved": "https://registry.npmjs.org/socket.io-client/-/socket.io-client-4.7.5.tgz", + "integrity": "sha512-sJ/tqHOCe7Z50JCBCXrsY3I2k03iOiUe+tj1OmKeD2lXPiGH/RUCdTZFoqVyN7l1MnpIzPrGtLcijffmeouNlQ==", + "license": "MIT", + "dependencies": { + "@socket.io/component-emitter": "~3.1.0", + "debug": "~4.3.2", + "engine.io-client": "~6.5.2", + "socket.io-parser": "~4.2.4" + }, + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/socket.io-parser": { + "version": "4.2.4", + "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-4.2.4.tgz", + "integrity": "sha512-/GbIKmo8ioc+NIWIhwdecY0ge+qVBSMdgxGygevmdHj24bsfgtCmcUUcQ5ZzcylGFHsN3k4HB4Cgkl96KVnuew==", + "license": "MIT", + "dependencies": { + "@socket.io/component-emitter": "~3.1.0", + "debug": "~4.3.1" + }, + "engines": { + "node": ">=10.0.0" + } + }, "node_modules/source-map-js": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz", @@ -4405,6 +4460,35 @@ "url": "https://github.com/chalk/strip-ansi?sponsor=1" } }, + "node_modules/ws": { + "version": "8.17.1", + "resolved": "https://registry.npmjs.org/ws/-/ws-8.17.1.tgz", + "integrity": "sha512-6XQFvXTkbfUOZOKKILFG1PDK2NDQs4azKQl26T0YS5CxqWLgXajbPZ+h4gZekJyRqFU8pvnbAbbs/3TgRPy+GQ==", + "license": "MIT", + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": ">=5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "utf-8-validate": { + "optional": true + } + } + }, + "node_modules/xmlhttprequest-ssl": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/xmlhttprequest-ssl/-/xmlhttprequest-ssl-2.0.0.tgz", + "integrity": "sha512-QKxVRxiRACQcVuQEYFsI1hhkrMlrXHPegbbd1yn9UHOmRxY+si12nQYzri3vbzt8VdTTRviqcKxcyllFas5z2A==", + "engines": { + "node": ">=0.4.0" + } + }, "node_modules/yaml": { "version": "1.10.2", "resolved": "https://registry.npmjs.org/yaml/-/yaml-1.10.2.tgz", diff --git a/app/package.json b/app/package.json index 8420c44..76ddc1a 100644 --- a/app/package.json +++ b/app/package.json @@ -35,6 +35,7 @@ }, "type": "module", "dependencies": { - "lucide-svelte": "^0.441.0" + "lucide-svelte": "^0.441.0", + "socket.io-client": "^4.7.5" } } diff --git a/app/src/lib/board.svelte b/app/src/lib/board.svelte index b597161..3d344aa 100644 --- a/app/src/lib/board.svelte +++ b/app/src/lib/board.svelte @@ -5,11 +5,11 @@ let { board, callback }: { board: Board; callback: (i: number, j: number) => void } = $props(); -
+
{#each board.board as row, i} {#each row as cell, j}

Opponent's Board

- gameState.opponentBoard.set(i, j, 'h')} - /> + gameState.attack(i, j)} />
@@ -50,9 +47,9 @@ type="text" bind:value={gameState.room} placeholder="Code" - class="input input-bordered w-full max-w-20" + class="input input-bordered w-full max-w-24 text-center" /> - + diff --git a/app/tailwind.config.js b/app/tailwind.config.js index 7a07c97..41ab34d 100644 --- a/app/tailwind.config.js +++ b/app/tailwind.config.js @@ -9,8 +9,8 @@ export default { ], daisyui: { - themes: ["nord"], // false: only light + dark | true: all themes | array: specific themes like this ["light", "dark", "cupcake"] - darkTheme: "nord", // name of one of the included themes for dark mode + themes: ["night"], // false: only light + dark | true: all themes | array: specific themes like this ["light", "dark", "cupcake"] + darkTheme: "night", // name of one of the included themes for dark mode base: true, // applies background color and foreground color for root element by default styled: true, // include daisyUI colors and design decisions for all components utils: true, // adds responsive and modifier utility classes diff --git a/build.rs b/build.rs new file mode 100644 index 0000000..d506869 --- /dev/null +++ b/build.rs @@ -0,0 +1,5 @@ +// generated by `sqlx migrate build-script` +fn main() { + // trigger recompilation when a new migration is added + println!("cargo:rerun-if-changed=migrations"); +} diff --git a/migrations/0001_battleship.sql b/migrations/0001_battleship.sql new file mode 100644 index 0000000..097bab8 --- /dev/null +++ b/migrations/0001_battleship.sql @@ -0,0 +1,28 @@ +-- DROP OWNED BY CURRENT_USER CASCADE; +CREATE TYPE STAT AS ENUM ('waiting', 'p1turn', 'p2turn'); + +CREATE TABLE IF NOT EXISTS players ( + id CHAR(16) PRIMARY KEY, + board CHAR(10) [10], + room_code CHAR(4) +); + +CREATE TABLE IF NOT EXISTS rooms ( + code CHAR(4) PRIMARY KEY, + player1_id CHAR(16), + player2_id CHAR(16), + stat STAT DEFAULT 'waiting' +); + +ALTER TABLE players +ADD CONSTRAINT fk_room_code FOREIGN KEY (room_code) REFERENCES rooms (code) ON DELETE +SET NULL; + +ALTER TABLE rooms +ADD CONSTRAINT fk_player1 FOREIGN KEY (player1_id) REFERENCES players (id) ON DELETE +SET NULL, + ADD CONSTRAINT fk_player2 FOREIGN KEY (player2_id) REFERENCES players (id) ON DELETE +SET NULL; + +CREATE INDEX idx_player_room_code ON players (room_code); +CREATE INDEX idx_room_status ON rooms (stat); \ No newline at end of file diff --git a/src/game.rs b/src/game.rs index 7782184..ef615a8 100644 --- a/src/game.rs +++ b/src/game.rs @@ -1,13 +1,190 @@ +use std::{collections::HashMap, sync::Arc}; + use axum::Json; use rand::Rng; -use serde::{Deserialize, Serialize}; +use serde::Deserialize; +use socketioxide::socket::Sid; +use tokio::sync::RwLock; -#[derive(Debug, Deserialize, Serialize)] +pub const ROOM_CODE_LENGTH: usize = 4; + +// #[derive(Default, Clone)] +// pub struct Store { +// rooms: Arc>>, +// sockets: Arc>>, +// } + +// impl Store { +// pub async fn add_room(&self, code: String) { +// let mut store = self.rooms.write().await; +// store.insert( +// code.clone(), +// Room { +// code, +// ..Default::default() +// }, +// ); +// } +pub async fn add_room(code: String, pool: &sqlx::PgPool) -> Result<(), sqlx::Error> { + sqlx::query!("INSERT INTO rooms (code) VALUES ($1)", code) + .execute(pool) + .await?; + Ok(()) +} + +pub async fn join_room(sid: Sid, code: String, pool: &sqlx::PgPool) -> Result<(), sqlx::Error> { + let room = sqlx::query!( + r#"SELECT player1_id, player2_id FROM rooms WHERE code = $1"#, + code + ) + .fetch_one(pool) + .await?; + + if room.player1_id.is_some() && room.player2_id.is_some() { + return Err(sqlx::Error::RowNotFound); // room full + } + + let mut txn = pool.begin().await?; + + sqlx::query!( + r#"INSERT INTO players (id, room_code) VALUES ($1, $2) ON CONFLICT (id) DO UPDATE SET room_code = $2"#, + sid.as_str(), + code + ) + .execute(&mut *txn) + .await?; + + sqlx::query(&format!( + "UPDATE rooms SET player{}_id = $1 WHERE code = $2", + if room.player1_id.is_none() { "1" } else { "2" } + )) + .bind(sid.as_str()) + .bind(code) + .execute(&mut *txn) + .await?; + + txn.commit().await?; + Ok(()) +} +// pub async fn join_room(&self, sid: Sid, code: String) -> Result<(), ()> { +// if self.rooms.read().await.get(&code).is_none() { +// return Err(()); +// }; +// let mut sockets = self.sockets.write().await; +// let player = sockets +// .entry(sid) +// .and_modify(|p| p.room = Some(code.clone())) +// .or_insert(Arc::new(Player { +// sid, +// room: Some(code.clone()), +// board: None, +// })); +// let mut rooms = self.rooms.write().await; +// let Some(room) = rooms.get_mut(&code) else { +// return Err(()); +// }; + +// if room.player1.is_none() { +// room.player1 = Some(Arc::clone(player)); +// } +// Ok(()) +// } + +// pub async fn add_board(&self, sid: Sid, board: Board) -> Result<(), ()> { +// let mut store = self.sockets.write().await; +// if let Some(player) = store.get_mut(&sid) { +// player.board = Some(board); +// } else { +// return Err(()); +// } +// Ok(()) +// } + +// pub async fn start(&self, code: String, sid: Sid) -> Result<(), ()> { +// let mut store = self.rooms.write().await; +// let Some(room) = store.get_mut(&code) else { +// return Err(()); +// }; +// dbg!(&room); +// let (Some(player1), Some(player2)) = (room.player1, room.player2) else { +// return Err(()); +// }; + +// if player1.sid == sid { +// room.status = Status::Player1Turn; +// } else if player2.sid == sid { +// room.status = Status::Player2Turn; +// } else { +// return Err(()); +// } +// Ok(()) +// } + +// pub async fn attack(&self, sid: Sid, (i, j): (usize, usize)) -> Result { +// let sockets = self.sockets.read().await; +// let Some(player) = sockets.get(&sid) else { +// return Err(()); +// }; +// let mut rooms = self.rooms.write().await; +// let Some(room) = rooms.get_mut(player.room.as_ref().unwrap()) else { +// return Err(()); +// }; + +// match room.status { +// Status::Player1Turn if player.sid == room.player1.as_ref().unwrap().sid => { +// room.status = Status::Player2Turn; +// return Ok(room.player2.as_ref().unwrap().board.as_ref().unwrap().0[i][j] == 's'); +// } +// Status::Player2Turn if player.sid == room.player2.as_ref().unwrap().sid => { +// room.status = Status::Player1Turn; +// return Ok(room.player1.as_ref().unwrap().board.as_ref().unwrap().0[i][j] == 's'); +// } +// _ => return Err(()), +// } + +// Err(()) +// } +// } + +// #[derive(Default, Debug)] +// struct Room { +// code: String, +// player1: Option>, +// player2: Option>, +// status: Status, +// } + +// #[derive(Debug)] +// struct Player { +// sid: Sid, +// board: Option, +// room: Option, +// } + +#[derive(Debug, sqlx::Type)] +#[sqlx(type_name = "STAT", rename_all = "lowercase")] +enum Status { + Waiting, + P1Turn, + P2Turn, +} + +// impl Default for Status { +// fn default() -> Self { +// Status::Waiting +// } +// } + +#[derive(Debug, Deserialize)] pub struct Board(pub [[char; 10]; 10]); impl Board { const SHIPS: [i32; 5] = [5, 4, 3, 3, 2]; + pub fn from_json(Json(board): Json) -> Self { + board + } + pub fn randomize() -> Self { let mut board = Board([['e'; 10]; 10]); for &length in Self::SHIPS.iter() { @@ -43,18 +220,14 @@ impl Board { false } - pub async fn from_json(Json(board): Json) -> Self { - board - } - - fn validate_syntax(&self) -> bool { - self.0 - .iter() - .all(|row| row.iter().all(|cell| matches!(cell, 'e' | 'h' | 'm' | 's'))) - } + // fn validate_syntax(&self) -> bool { + // self.0 + // .iter() + // .all(|row| row.iter().all(|cell| matches!(cell, 'e' | 'h' | 'm' | 's'))) + // } } -pub async fn create_board_route(board: Json) -> String { - let board = Board::from_json(board).await; - format!("{:?}", board) -} +// pub async fn create_board_route(board: Json) -> Json { +// let board = Board::from_json(board).await; +// Json(format!("{:?}", board.0)) +// } diff --git a/src/main.rs b/src/main.rs index bf63530..04ef94f 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,18 +1,135 @@ mod game; +use std::{str::FromStr, sync::Arc}; -use axum::{ - routing::{get, post}, - Json, Router, +use axum::Router; +use dotenv::dotenv; +use futures_util::stream::StreamExt; +use game::{join_room, Board, ROOM_CODE_LENGTH}; +use rand::Rng; +use serde_json::Value; +use socketioxide::{ + adapter::Room, + extract::{AckSender, Data, SocketRef, State}, + socket::Sid, + SocketIo, }; -use game::Board; -use serde::Serialize; use tokio::net::TcpListener; +use tower_http::cors::CorsLayer; +use tracing_subscriber::FmtSubscriber; #[tokio::main] -async fn main() { - let app = Router::new().route("/", post(game::create_board_route)); +async fn main() -> Result<(), Box> { + tracing::subscriber::set_global_default( + FmtSubscriber::builder() + .with_max_level(tracing::Level::INFO) + .finish(), + )?; + let _ = dotenv(); + let url = std::env::var("DATABASE_URL")?; + let pool = sqlx::postgres::PgPool::connect(&url).await?; + sqlx::migrate!("./migrations").run(&pool).await?; + join_room( + Sid::from_str("aaaaaaaaaaaaaaaa").unwrap(), + "AAAB".to_string(), + &pool, + ) + .await?; + let (layer, io) = SocketIo::builder().with_state(pool).build_layer(); + // io.ns("/", on_connect); + let app = Router::new() + // .route("/", post(game::create_board_route)) + .layer(layer) + .layer(CorsLayer::very_permissive()); - let listener = TcpListener::bind("127.0.0.1:3000").await.unwrap(); - println!("listening on {}", listener.local_addr().unwrap()); - axum::serve(listener, app).await.unwrap(); + let listener = TcpListener::bind("127.0.0.1:3000").await?; + println!("listening on {}", listener.local_addr()?); + axum::serve(listener, app).await?; + Ok(()) } + +// fn on_connect(socket: SocketRef, io: SocketIo) { +// tracing::info!("Connected: {:?}", socket.id); +// // tracing::info!( +// // "All rooms and sockets: {:?}", +// // io.rooms() +// // .unwrap() +// // .iter() +// // .map(|room| { (room, io.within(room.clone()).sockets().unwrap()) }) +// // ); + +// socket.on( +// "create", +// |socket: SocketRef, store: State| async move { +// if !socket.rooms().unwrap().is_empty() { +// socket +// .emit("created-room", socket.rooms().unwrap().first()) +// .unwrap(); +// println!("{} Already in a room", socket.id); +// return; +// } + +// let room: String = rand::thread_rng() +// .sample_iter(&rand::distributions::Alphanumeric) +// .take(ROOM_CODE_LENGTH) +// .map(|x| char::to_ascii_uppercase(&(x as char))) +// .collect(); +// tracing::info!("Creating room: {:?}", room); +// store.add_room(room.clone()).await; +// store.join_room(socket.id, room.clone()).await.unwrap(); +// socket.leave_all().unwrap(); +// socket.join(room.clone()).unwrap(); +// socket.emit("created-room", &room).unwrap(); +// }, +// ); + +// socket.on( +// "join", +// |socket: SocketRef, Data::(room), store: State| async move { +// if room.len() != ROOM_CODE_LENGTH { +// return; +// } +// tracing::info!("Joining room: {:?}", room); +// store.join_room(socket.id, room.clone()).await.unwrap(); +// socket.leave_all().unwrap(); +// socket.join(room.clone()).unwrap(); +// if socket.within(room.clone()).sockets().unwrap().len() != 2 { +// return; +// } +// let ack_stream = socket +// .within(room.clone()) +// .emit_with_ack::>("upload", ()) +// .unwrap(); +// ack_stream +// .for_each(|(id, ack)| { +// let store = store.clone(); +// async move { +// match ack { +// Ok(mut ack) => { +// store.add_board(id, ack.data.pop().unwrap()).await.unwrap(); +// } +// Err(err) => tracing::error!("Ack error, {}", err), +// } +// } +// }) +// .await; +// store.start(room.clone(), socket.id).await.unwrap(); +// tracing::info!("Game started"); +// socket.within(room.clone()).emit("turn", socket.id).unwrap(); +// }, +// ); + +// socket.on( +// "attack", +// |socket: SocketRef, Data::<[usize; 2]>([i, j]), ack: AckSender, store: State| async move { +// let res = store.attack(socket.id, (i, j)).await.unwrap(); +// tracing::info!("Attacking at: ({}, {}), result: {}", i, j, res); +// ack.send(res).unwrap(); +// }, +// ); + +// socket.on_disconnect(|socket: SocketRef, store: State| { +// tracing::info!("Disconnecting: {:?}", socket.id); +// socket.leave_all().unwrap(); +// // TODO: Delete room +// }); +// }