diff --git a/.vscode/extensions.json b/.vscode/extensions.json index 24d7cc6..7ed67a1 100644 --- a/.vscode/extensions.json +++ b/.vscode/extensions.json @@ -1,3 +1,9 @@ { - "recommendations": ["tauri-apps.tauri-vscode", "rust-lang.rust-analyzer"] + "recommendations": [ + "tauri-apps.tauri-vscode", + "rust-lang.rust-analyzer", + "biomejs.biome", + "oven.bun-vscode", + "streetsidesoftware.code-spell-checker" + ] } diff --git a/.vscode/settings.json b/.vscode/settings.json index 2a0d05c..4c3d6b1 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -20,5 +20,5 @@ "autorun", "blueprintjs", "tanstack" - ] + ], } \ No newline at end of file diff --git a/bun.lockb b/bun.lockb old mode 100644 new mode 100755 index f32ead9..17ffac2 Binary files a/bun.lockb and b/bun.lockb differ diff --git a/package.json b/package.json index b23f25b..d7954c3 100644 --- a/package.json +++ b/package.json @@ -1,11 +1,11 @@ { "name": "database-browser", "private": true, - "version": "0.0.3", + "version": "0.0.4", "type": "module", "scripts": { - "dev": "vite", - "build": "tsc && vite build", + "dev": "vite dev", + "build": "vite build", "preview": "vite preview", "check": "bunx tsc --noEmit", "tauri": "tauri", @@ -18,11 +18,11 @@ "@tanstack/react-query": "^5.56.2", "@tanstack/react-query-devtools": "^5.56.2", "@tanstack/react-router": "^1.58.3", - "@tauri-apps/api": "^1.6.0", + "@tauri-apps/api": "^2.0.0-rc.0", + "@tauri-apps/plugin-fs": "~2.0.0-rc", + "@tauri-apps/plugin-shell": "~2.0.0-rc", "@xyflow/react": "^12.3.0", "clsx": "^2.1.1", - "d3-force": "^3.0.0", - "d3-quadtree": "^3.0.1", "date-fns": "^4.1.0", "jotai": "^2.10.0", "mobx": "^6.13.2", @@ -41,14 +41,14 @@ "@biomejs/biome": "1.9.2", "@tanstack/router-devtools": "^1.58.3", "@tanstack/router-plugin": "^1.58.4", - "@tauri-apps/cli": "^1", - "@types/d3-force": "^3.0.10", - "@types/d3-quadtree": "^3.0.6", + "@tauri-apps/cli": "^2.0.0-rc.18", + "@types/bun": "^1.1.10", "@types/react": "^18.2.15", "@types/react-dom": "^18.2.7", "@vitejs/plugin-react": "^4.2.1", "autoprefixer": "^10.4.20", "browserslist": "^4.24.0", + "concurrently": "^9.0.1", "postcss": "^8.4.47", "tailwindcss": "^3.4.12", "typescript": "^5.2.2", diff --git a/src-tauri/Cargo.lock b/src-tauri/Cargo.lock index af1fd9e..0ed396c 100644 --- a/src-tauri/Cargo.lock +++ b/src-tauri/Cargo.lock @@ -81,26 +81,25 @@ dependencies = [ [[package]] name = "atk" -version = "0.15.1" +version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2c3d816ce6f0e2909a96830d6911c2aff044370b1ef92d7f267b43bae5addedd" +checksum = "b4af014b17dd80e8af9fa689b2d4a211ddba6eb583c1622f35d0cb543f6b17e4" dependencies = [ "atk-sys", - "bitflags 1.3.2", "glib", "libc", ] [[package]] name = "atk-sys" -version = "0.15.1" +version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "58aeb089fb698e06db8089971c7ee317ab9644bade33383f63631437b03aafb6" +checksum = "251e0b7d90e33e0ba930891a505a9a35ece37b2dd37a14f3ffc306c13b980009" dependencies = [ "glib-sys", "gobject-sys", "libc", - "system-deps 6.2.2", + "system-deps", ] [[package]] @@ -124,12 +123,6 @@ dependencies = [ "windows-targets 0.52.6", ] -[[package]] -name = "base64" -version = "0.13.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9e1b586273c5702936fe7b7d6896644d8be71e6314cfe09d3167c95f712589e8" - [[package]] name = "base64" version = "0.21.7" @@ -153,6 +146,9 @@ name = "bitflags" version = "2.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b048fb63fd8b5923fc5aa7b340d8e156aec7ec02f0c78fa8a6ddc2613f6f71de" +dependencies = [ + "serde", +] [[package]] name = "block" @@ -169,6 +165,15 @@ dependencies = [ "generic-array", ] +[[package]] +name = "block2" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2c132eebf10f5cad5289222520a4a058514204aed6d791f1cf4fe8088b82d15f" +dependencies = [ + "objc2", +] + [[package]] name = "brotli" version = "6.0.0" @@ -190,16 +195,6 @@ dependencies = [ "alloc-stdlib", ] -[[package]] -name = "bstr" -version = "1.10.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "40723b8fb387abc38f4f4a37c09073622e41dd12327033091ef8950659e6dc0c" -dependencies = [ - "memchr", - "serde", -] - [[package]] name = "bumpalo" version = "3.16.0" @@ -223,39 +218,75 @@ name = "bytes" version = "1.7.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "428d9aa8fbc0670b7b8d6030a7fadd0f86151cae55e4dbbece15f3780a3dfaf3" +dependencies = [ + "serde", +] [[package]] name = "cairo-rs" -version = "0.15.12" +version = "0.18.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c76ee391b03d35510d9fa917357c7f1855bd9a6659c95a1b392e33f49b3369bc" +checksum = "8ca26ef0159422fb77631dc9d17b102f253b876fe1586b03b803e63a309b4ee2" dependencies = [ - "bitflags 1.3.2", + "bitflags 2.6.0", "cairo-sys-rs", "glib", "libc", + "once_cell", "thiserror", ] [[package]] name = "cairo-sys-rs" -version = "0.15.1" +version = "0.18.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3c55d429bef56ac9172d25fecb85dc8068307d17acd74b377866b7a1ef25d3c8" +checksum = "685c9fa8e590b8b3d678873528d83411db17242a73fccaed827770ea0fedda51" dependencies = [ "glib-sys", "libc", - "system-deps 6.2.2", + "system-deps", +] + +[[package]] +name = "camino" +version = "1.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b96ec4966b5813e2c0507c1f86115c8c5abaadc3980879c3424042a02fd1ad3" +dependencies = [ + "serde", +] + +[[package]] +name = "cargo-platform" +version = "0.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "24b1f0365a6c6bb4020cd05806fd0d33c44d38046b8bd7f0e40814b9763cabfc" +dependencies = [ + "serde", +] + +[[package]] +name = "cargo_metadata" +version = "0.18.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2d886547e41f740c616ae73108f6eb70afe6d940c7bc697cb30f13daec073037" +dependencies = [ + "camino", + "cargo-platform", + "semver", + "serde", + "serde_json", + "thiserror", ] [[package]] name = "cargo_toml" -version = "0.15.3" +version = "0.17.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "599aa35200ffff8f04c1925aa1acc92fa2e08874379ef42e210a80e527e60838" +checksum = "8a969e13a7589e9e3e4207e153bae624ade2b5622fb4684a4923b23ec3d57719" dependencies = [ "serde", - "toml 0.7.8", + "toml 0.8.19", ] [[package]] @@ -284,15 +315,6 @@ dependencies = [ "uuid", ] -[[package]] -name = "cfg-expr" -version = "0.9.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3431df59f28accaf4cb4eed4a9acc66bea3f3c3753aa6cdc2f024174ef232af7" -dependencies = [ - "smallvec", -] - [[package]] name = "cfg-expr" version = "0.15.8" @@ -309,6 +331,12 @@ version = "1.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" +[[package]] +name = "cfg_aliases" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "613afe47fcd5fac7ccf1db93babcb082c5994d996f20b8b159f2ad1658eb5724" + [[package]] name = "chrono" version = "0.4.38" @@ -326,11 +354,11 @@ dependencies = [ [[package]] name = "cocoa" -version = "0.24.1" +version = "0.26.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f425db7937052c684daec3bd6375c8abe2d146dca4b8b143d6db777c39138f3a" +checksum = "f79398230a6e2c08f5c9760610eb6924b52aa9e7950a619602baba59dcbbdbb2" dependencies = [ - "bitflags 1.3.2", + "bitflags 2.6.0", "block", "cocoa-foundation", "core-foundation", @@ -342,11 +370,11 @@ dependencies = [ [[package]] name = "cocoa-foundation" -version = "0.1.2" +version = "0.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8c6234cbb2e4c785b456c0644748b1ac416dd045799740356f8363dfe00c93f7" +checksum = "e14045fb83be07b5acf1c0884b2180461635b433455fa35d1cd6f17f1450679d" dependencies = [ - "bitflags 1.3.2", + "bitflags 2.6.0", "block", "core-foundation", "core-graphics-types", @@ -354,12 +382,6 @@ dependencies = [ "objc", ] -[[package]] -name = "color_quant" -version = "1.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3d7b894f5411737b7867f4827955924d7c254fc9f4d91a6aad6b097804b1018b" - [[package]] name = "combine" version = "4.6.7" @@ -378,9 +400,9 @@ checksum = "6245d59a3e82a7fc217c5828a6692dbc6dfb63a0c8c90495621f7b9d79704a0e" [[package]] name = "core-foundation" -version = "0.9.4" +version = "0.10.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "91e195e091a93c46f7102ec7818a2aa394e1e1771c3ab4825963fa03e45afb8f" +checksum = "b55271e5c8c478ad3f38ad24ef34923091e0548492a266d19b3c0b4d82574c63" dependencies = [ "core-foundation-sys", "libc", @@ -394,11 +416,11 @@ checksum = "773648b94d0e5d620f64f280777445740e61fe701025087ec8b57f45c791888b" [[package]] name = "core-graphics" -version = "0.22.3" +version = "0.24.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2581bbab3b8ffc6fcbd550bf46c355135d16e9ff2a6ea032ad6b9bf1d7efe4fb" +checksum = "fa95a34622365fa5bbf40b20b75dba8dfa8c94c734aea8ac9a5ca38af14316f1" dependencies = [ - "bitflags 1.3.2", + "bitflags 2.6.0", "core-foundation", "core-graphics-types", "foreign-types", @@ -407,11 +429,11 @@ dependencies = [ [[package]] name = "core-graphics-types" -version = "0.1.3" +version = "0.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "45390e6114f68f718cc7a830514a96f903cccd70d02a8f6d9f643ac4ba45afaf" +checksum = "3d44a101f213f6c4cdc1853d4b78aef6db6bdfa3468798cc1d9912f4735013eb" dependencies = [ - "bitflags 1.3.2", + "bitflags 2.6.0", "core-foundation", "libc", ] @@ -443,25 +465,6 @@ dependencies = [ "crossbeam-utils", ] -[[package]] -name = "crossbeam-deque" -version = "0.8.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "613f8cc01fe9cf1a3eb3d7f488fd2fa8388403e97039e2f73692932e291a770d" -dependencies = [ - "crossbeam-epoch", - "crossbeam-utils", -] - -[[package]] -name = "crossbeam-epoch" -version = "0.9.18" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5b82ac4a3c2ca9c3460964f020e1402edd5753411d7737aa39c3714ad1b5420e" -dependencies = [ - "crossbeam-utils", -] - [[package]] name = "crossbeam-utils" version = "0.8.20" @@ -560,6 +563,8 @@ dependencies = [ "serde_json", "tauri", "tauri-build", + "tauri-plugin-fs", + "tauri-plugin-shell", "tokio", "tokio-postgres", ] @@ -599,24 +604,24 @@ dependencies = [ ] [[package]] -name = "dirs-next" -version = "2.0.0" +name = "dirs" +version = "5.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b98cf8ebf19c3d1b223e151f99a4f9f0690dca41414773390fc824184ac833e1" +checksum = "44c45a9d03d6676652bcb5e724c7e988de1acad23a711b5217ab9cbecbec2225" dependencies = [ - "cfg-if", - "dirs-sys-next", + "dirs-sys", ] [[package]] -name = "dirs-sys-next" -version = "0.1.2" +name = "dirs-sys" +version = "0.4.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4ebda144c4fe02d1f7ea1a7d9641b6fc6b580adcfa024ae48797ecdeb6825b4d" +checksum = "520f05a5cbd335fae5a99ff7a6ab8627577660ee5cfd6a94a6a929b52ff0321c" dependencies = [ "libc", + "option-ext", "redox_users", - "winapi", + "windows-sys 0.48.0", ] [[package]] @@ -625,6 +630,38 @@ version = "0.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "bd0c93bb4b0c6d9b77f4435b0ae98c24d17f1c45b2ff844c6151a07256ca923b" +[[package]] +name = "dlopen2" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9e1297103d2bbaea85724fcee6294c2d50b1081f9ad47d0f6f6f61eda65315a6" +dependencies = [ + "dlopen2_derive", + "libc", + "once_cell", + "winapi", +] + +[[package]] +name = "dlopen2_derive" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f2b99bf03862d7f545ebc28ddd33a665b50865f4dfd84031a393823879bd4c54" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.77", +] + +[[package]] +name = "dpi" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f25c0e292a7ca6d6498557ff1df68f32c99850012b6ea401cf8daf771f22ff53" +dependencies = [ + "serde", +] + [[package]] name = "dtoa" version = "1.0.9" @@ -646,6 +683,12 @@ version = "1.0.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "92773504d58c093f6de2459af4af33faa518c13451eb8f2b5698ed3d36e7c813" +[[package]] +name = "dyn-clone" +version = "1.0.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0d6ef0072f8a535281e4876be788938b528e9a1d43900b82c2569af7da799125" + [[package]] name = "embed-resource" version = "2.4.3" @@ -682,13 +725,13 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5" [[package]] -name = "errno" -version = "0.3.9" +name = "erased-serde" +version = "0.4.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "534c5cf6194dfab3db3242765c03bbe257cf92f22b38f6bc0c58d59108a820ba" +checksum = "24e2389d65ab4fab27dc2a5de7b191e1f6617d1f1c8855c0dc569c94a4cbb18d" dependencies = [ - "libc", - "windows-sys 0.52.0", + "serde", + "typeid", ] [[package]] @@ -697,12 +740,6 @@ version = "0.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "4443176a9f2c162692bd3d352d745ef9413eec5782a80d8fd6f8a1ac692a07f7" -[[package]] -name = "fastrand" -version = "2.1.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e8c02a5121d4ea3eb16a80748c74f5549a5665e4c21333c6098f283870fbdea6" - [[package]] name = "fdeflate" version = "0.3.4" @@ -722,18 +759,6 @@ dependencies = [ "rustc_version", ] -[[package]] -name = "filetime" -version = "0.2.25" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "35c0522e981e68cbfa8c3f978441a5f34b30b96e146b33cd3359176b50fe8586" -dependencies = [ - "cfg-if", - "libc", - "libredox", - "windows-sys 0.59.0", -] - [[package]] name = "flate2" version = "1.0.33" @@ -761,18 +786,30 @@ checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" [[package]] name = "foreign-types" -version = "0.3.2" +version = "0.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f6f339eb8adc052cd2ca78910fda869aefa38d22d5cb648e6485e4d3fc06f3b1" +checksum = "d737d9aa519fb7b749cbc3b962edcf310a8dd1f4b67c91c4f83975dbdd17d965" dependencies = [ + "foreign-types-macros", "foreign-types-shared", ] +[[package]] +name = "foreign-types-macros" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1a5c6c585bc94aaf2c7b51dd4c2ba22680844aba4c687be581871a6f518c5742" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.77", +] + [[package]] name = "foreign-types-shared" -version = "0.1.1" +version = "0.3.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "00b0228411908ca8685dba7fc2cdd70ec9990a6e753e89b6ac91a84c40fbaf4b" +checksum = "aa9a19cbb55df58761df49b23516a86d432839add4af60fc256da840f66ed35b" [[package]] name = "form_urlencoded" @@ -893,11 +930,10 @@ dependencies = [ [[package]] name = "gdk" -version = "0.15.4" +version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a6e05c1f572ab0e1f15be94217f0dc29088c248b14f792a5ff0af0d84bcda9e8" +checksum = "f5ba081bdef3b75ebcdbfc953699ed2d7417d6bd853347a42a37d76406a33646" dependencies = [ - "bitflags 1.3.2", "cairo-rs", "gdk-pixbuf", "gdk-sys", @@ -909,35 +945,35 @@ dependencies = [ [[package]] name = "gdk-pixbuf" -version = "0.15.11" +version = "0.18.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ad38dd9cc8b099cceecdf41375bb6d481b1b5a7cd5cd603e10a69a9383f8619a" +checksum = "50e1f5f1b0bfb830d6ccc8066d18db35c487b1b2b1e8589b5dfe9f07e8defaec" dependencies = [ - "bitflags 1.3.2", "gdk-pixbuf-sys", "gio", "glib", "libc", + "once_cell", ] [[package]] name = "gdk-pixbuf-sys" -version = "0.15.10" +version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "140b2f5378256527150350a8346dbdb08fadc13453a7a2d73aecd5fab3c402a7" +checksum = "3f9839ea644ed9c97a34d129ad56d38a25e6756f99f3a88e15cd39c20629caf7" dependencies = [ "gio-sys", "glib-sys", "gobject-sys", "libc", - "system-deps 6.2.2", + "system-deps", ] [[package]] name = "gdk-sys" -version = "0.15.1" +version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "32e7a08c1e8f06f4177fb7e51a777b8c1689f743a7bc11ea91d44d2226073a88" +checksum = "31ff856cb3386dae1703a920f803abafcc580e9b5f711ca62ed1620c25b51ff2" dependencies = [ "cairo-sys-rs", "gdk-pixbuf-sys", @@ -947,47 +983,48 @@ dependencies = [ "libc", "pango-sys", "pkg-config", - "system-deps 6.2.2", + "system-deps", ] [[package]] name = "gdkwayland-sys" -version = "0.15.3" +version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cca49a59ad8cfdf36ef7330fe7bdfbe1d34323220cc16a0de2679ee773aee2c2" +checksum = "a90fbf5c033c65d93792192a49a8efb5bb1e640c419682a58bb96f5ae77f3d4a" dependencies = [ "gdk-sys", "glib-sys", "gobject-sys", "libc", "pkg-config", - "system-deps 6.2.2", + "system-deps", ] [[package]] -name = "gdkx11-sys" -version = "0.15.1" +name = "gdkx11" +version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b4b7f8c7a84b407aa9b143877e267e848ff34106578b64d1e0a24bf550716178" +checksum = "db2ea8a4909d530f79921290389cbd7c34cb9d623bfe970eaae65ca5f9cd9cce" dependencies = [ - "gdk-sys", - "glib-sys", + "gdk", + "gdkx11-sys", + "gio", + "glib", "libc", - "system-deps 6.2.2", "x11", ] [[package]] -name = "generator" -version = "0.7.5" +name = "gdkx11-sys" +version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5cc16584ff22b460a382b7feec54b23d2908d858152e5739a120b949293bd74e" +checksum = "fee8f00f4ee46cad2939b8990f5c70c94ff882c3028f3cc5abf950fa4ab53043" dependencies = [ - "cc", + "gdk-sys", + "glib-sys", "libc", - "log", - "rustversion", - "windows 0.48.0", + "system-deps", + "x11", ] [[package]] @@ -1030,49 +1067,54 @@ checksum = "32085ea23f3234fc7846555e85283ba4de91e21016dc0455a16286d87a292d64" [[package]] name = "gio" -version = "0.15.12" +version = "0.18.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "68fdbc90312d462781a395f7a16d96a2b379bb6ef8cd6310a2df272771c4283b" +checksum = "d4fc8f532f87b79cbc51a79748f16a6828fb784be93145a322fa14d06d354c73" dependencies = [ - "bitflags 1.3.2", "futures-channel", "futures-core", "futures-io", + "futures-util", "gio-sys", "glib", "libc", "once_cell", + "pin-project-lite", + "smallvec", "thiserror", ] [[package]] name = "gio-sys" -version = "0.15.10" +version = "0.18.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "32157a475271e2c4a023382e9cab31c4584ee30a97da41d3c4e9fdd605abcf8d" +checksum = "37566df850baf5e4cb0dfb78af2e4b9898d817ed9263d1090a2df958c64737d2" dependencies = [ "glib-sys", "gobject-sys", "libc", - "system-deps 6.2.2", + "system-deps", "winapi", ] [[package]] name = "glib" -version = "0.15.12" +version = "0.18.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "edb0306fbad0ab5428b0ca674a23893db909a98582969c9b537be4ced78c505d" +checksum = "233daaf6e83ae6a12a52055f568f9d7cf4671dabb78ff9560ab6da230ce00ee5" dependencies = [ - "bitflags 1.3.2", + "bitflags 2.6.0", "futures-channel", "futures-core", "futures-executor", "futures-task", + "futures-util", + "gio-sys", "glib-macros", "glib-sys", "gobject-sys", "libc", + "memchr", "once_cell", "smallvec", "thiserror", @@ -1080,27 +1122,26 @@ dependencies = [ [[package]] name = "glib-macros" -version = "0.15.13" +version = "0.18.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "10c6ae9f6fa26f4fb2ac16b528d138d971ead56141de489f8111e259b9df3c4a" +checksum = "0bb0228f477c0900c880fd78c8759b95c7636dbd7842707f49e132378aa2acdc" dependencies = [ - "anyhow", "heck 0.4.1", - "proc-macro-crate", + "proc-macro-crate 2.0.0", "proc-macro-error", "proc-macro2", "quote", - "syn 1.0.109", + "syn 2.0.77", ] [[package]] name = "glib-sys" -version = "0.15.10" +version = "0.18.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ef4b192f8e65e9cf76cbf4ea71fa8e3be4a0e18ffe3d68b8da6836974cc5bad4" +checksum = "063ce2eb6a8d0ea93d2bf8ba1957e78dbab6be1c2220dd3daca57d5a9d869898" dependencies = [ "libc", - "system-deps 6.2.2", + "system-deps", ] [[package]] @@ -1109,38 +1150,24 @@ version = "0.3.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d2fabcfbdc87f4758337ca535fb41a6d701b65693ce38287d856d1674551ec9b" -[[package]] -name = "globset" -version = "0.4.15" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "15f1ce686646e7f1e19bf7d5533fe443a45dbfb990e00629110797578b42fb19" -dependencies = [ - "aho-corasick", - "bstr", - "log", - "regex-automata 0.4.7", - "regex-syntax 0.8.4", -] - [[package]] name = "gobject-sys" -version = "0.15.10" +version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0d57ce44246becd17153bd035ab4d32cfee096a657fc01f2231c9278378d1e0a" +checksum = "0850127b514d1c4a4654ead6dedadb18198999985908e6ffe4436f53c785ce44" dependencies = [ "glib-sys", "libc", - "system-deps 6.2.2", + "system-deps", ] [[package]] name = "gtk" -version = "0.15.5" +version = "0.18.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "92e3004a2d5d6d8b5057d2b57b3712c9529b62e82c77f25c1fecde1fd5c23bd0" +checksum = "93c4f5e0e20b60e10631a5f06da7fe3dda744b05ad0ea71fee2f47adf865890c" dependencies = [ "atk", - "bitflags 1.3.2", "cairo-rs", "field-offset", "futures-channel", @@ -1151,16 +1178,15 @@ dependencies = [ "gtk-sys", "gtk3-macros", "libc", - "once_cell", "pango", "pkg-config", ] [[package]] name = "gtk-sys" -version = "0.15.3" +version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d5bc2f0587cba247f60246a0ca11fe25fb733eabc3de12d1965fc07efab87c84" +checksum = "771437bf1de2c1c0b496c11505bdf748e26066bbe942dfc8f614c9460f6d7722" dependencies = [ "atk-sys", "cairo-sys-rs", @@ -1171,21 +1197,20 @@ dependencies = [ "gobject-sys", "libc", "pango-sys", - "system-deps 6.2.2", + "system-deps", ] [[package]] name = "gtk3-macros" -version = "0.15.6" +version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "684c0456c086e8e7e9af73ec5b84e35938df394712054550e81558d21c44ab0d" +checksum = "c6063efb63db582968fb7df72e1ae68aa6360dcfb0a75143f34fc7d616bad75e" dependencies = [ - "anyhow", - "proc-macro-crate", + "proc-macro-crate 1.3.1", "proc-macro-error", "proc-macro2", "quote", - "syn 1.0.109", + "syn 2.0.77", ] [[package]] @@ -1200,15 +1225,6 @@ version = "0.14.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e5274423e17b7c9fc20b6e7e208532f9b19825d82dfd615708b70edd83df41f1" -[[package]] -name = "heck" -version = "0.3.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6d621efb26863f0e9924c6ac577e8275e5e6b77455db64ffa6c65c904e9e132c" -dependencies = [ - "unicode-segmentation", -] - [[package]] name = "heck" version = "0.4.1" @@ -1258,9 +1274,9 @@ dependencies = [ [[package]] name = "http" -version = "0.2.12" +version = "1.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "601cbb57e577e2f5ef5be8e7b83f0f63994f25aa94d673e54a92d5c516d101f1" +checksum = "21b9ddb458710bc376481b842f5da65cdf31522de232c1ca8146abce2a358258" dependencies = [ "bytes", "fnv", @@ -1268,86 +1284,119 @@ dependencies = [ ] [[package]] -name = "http-range" -version = "0.1.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "21dec9db110f5f872ed9699c3ecf50cf16f423502706ba5c72462e28d3157573" - -[[package]] -name = "iana-time-zone" -version = "0.1.61" +name = "http-body" +version = "1.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "235e081f3925a06703c2d0117ea8b91f042756fd6e7a6e5d901e8ca1a996b220" +checksum = "1efedce1fb8e6913f23e0c92de8e62cd5b772a67e7b3946df930a62566c93184" dependencies = [ - "android_system_properties", - "core-foundation-sys", - "iana-time-zone-haiku", - "js-sys", - "wasm-bindgen", - "windows-core", + "bytes", + "http", ] [[package]] -name = "iana-time-zone-haiku" +name = "http-body-util" version = "0.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f31827a206f56af32e590ba56d5d2d085f558508192593743f16b2306495269f" +checksum = "793429d76616a256bcb62c2a2ec2bed781c8307e797e2598c50010f2bee2544f" dependencies = [ - "cc", + "bytes", + "futures-util", + "http", + "http-body", + "pin-project-lite", ] [[package]] -name = "ico" -version = "0.3.0" +name = "httparse" +version = "1.9.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e3804960be0bb5e4edb1e1ad67afd321a9ecfd875c3e65c099468fd2717d7cae" -dependencies = [ - "byteorder", - "png", -] +checksum = "7d71d3574edd2771538b901e6549113b4006ece66150fb69c0fb6d9a2adae946" [[package]] -name = "ident_case" -version = "1.0.1" +name = "hyper" +version = "1.4.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b9e0384b61958566e926dc50660321d12159025e767c18e043daf26b70104c39" - -[[package]] -name = "idna" -version = "0.5.0" +checksum = "50dfd22e0e76d0f662d429a5f80fcaf3855009297eab6a0a9f8543834744ba05" +dependencies = [ + "bytes", + "futures-channel", + "futures-util", + "http", + "http-body", + "httparse", + "itoa 1.0.11", + "pin-project-lite", + "smallvec", + "tokio", + "want", +] + +[[package]] +name = "hyper-util" +version = "0.1.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "634d9b1461af396cad843f47fdba5597a4f9e6ddd4bfb6ff5d85028c25cb12f6" +checksum = "41296eb09f183ac68eec06e03cdbea2e759633d4067b2f6552fc2e009bcad08b" dependencies = [ - "unicode-bidi", - "unicode-normalization", + "bytes", + "futures-channel", + "futures-util", + "http", + "http-body", + "hyper", + "pin-project-lite", + "socket2", + "tokio", + "tower-service", + "tracing", ] [[package]] -name = "ignore" -version = "0.4.23" +name = "iana-time-zone" +version = "0.1.61" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6d89fd380afde86567dfba715db065673989d6253f42b88179abd3eae47bda4b" +checksum = "235e081f3925a06703c2d0117ea8b91f042756fd6e7a6e5d901e8ca1a996b220" dependencies = [ - "crossbeam-deque", - "globset", - "log", - "memchr", - "regex-automata 0.4.7", - "same-file", - "walkdir", - "winapi-util", + "android_system_properties", + "core-foundation-sys", + "iana-time-zone-haiku", + "js-sys", + "wasm-bindgen", + "windows-core 0.52.0", ] [[package]] -name = "image" -version = "0.24.9" +name = "iana-time-zone-haiku" +version = "0.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5690139d2f55868e080017335e4b94cb7414274c74f1669c84fb5feba2c9f69d" +checksum = "f31827a206f56af32e590ba56d5d2d085f558508192593743f16b2306495269f" +dependencies = [ + "cc", +] + +[[package]] +name = "ico" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3804960be0bb5e4edb1e1ad67afd321a9ecfd875c3e65c099468fd2717d7cae" dependencies = [ - "bytemuck", "byteorder", - "color_quant", - "num-traits", + "png", +] + +[[package]] +name = "ident_case" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b9e0384b61958566e926dc50660321d12159025e767c18e043daf26b70104c39" + +[[package]] +name = "idna" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "634d9b1461af396cad843f47fdba5597a4f9e6ddd4bfb6ff5d85028c25cb12f6" +dependencies = [ + "unicode-bidi", + "unicode-normalization", ] [[package]] @@ -1374,9 +1423,9 @@ dependencies = [ [[package]] name = "infer" -version = "0.13.0" +version = "0.16.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f551f8c3a39f68f986517db0d1759de85881894fdc7db798bd2a9df9cb04b7fc" +checksum = "bc150e5ce2330295b8616ce0e3f53250e53af31759a9dbedad1621ba29151847" dependencies = [ "cfb", ] @@ -1390,6 +1439,31 @@ dependencies = [ "cfg-if", ] +[[package]] +name = "ipnet" +version = "2.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "187674a687eed5fe42285b40c6291f9a01517d415fad1c3cbc6a9f778af7fcd4" + +[[package]] +name = "is-docker" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "928bae27f42bc99b60d9ac7334e3a21d10ad8f1835a4e12ec3ec0464765ed1b3" +dependencies = [ + "once_cell", +] + +[[package]] +name = "is-wsl" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "173609498df190136aa7dea1a91db051746d339e18476eed5ca40521f02d7aa5" +dependencies = [ + "is-docker", + "once_cell", +] + [[package]] name = "itoa" version = "0.4.8" @@ -1404,9 +1478,9 @@ checksum = "49f1f14873335454500d59611f1cf4a4b0f786f9ac11f4312a78e4cf2566695b" [[package]] name = "javascriptcore-rs" -version = "0.16.0" +version = "1.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bf053e7843f2812ff03ef5afe34bb9c06ffee120385caad4f6b9967fcd37d41c" +checksum = "ca5671e9ffce8ffba57afc24070e906da7fc4b1ba66f2cabebf61bf2ea257fcc" dependencies = [ "bitflags 1.3.2", "glib", @@ -1415,28 +1489,30 @@ dependencies = [ [[package]] name = "javascriptcore-rs-sys" -version = "0.4.0" +version = "1.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "905fbb87419c5cde6e3269537e4ea7d46431f3008c5d057e915ef3f115e7793c" +checksum = "af1be78d14ffa4b75b66df31840478fef72b51f8c2465d4ca7c194da9f7a5124" dependencies = [ "glib-sys", "gobject-sys", "libc", - "system-deps 5.0.0", + "system-deps", ] [[package]] name = "jni" -version = "0.20.0" +version = "0.21.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "039022cdf4d7b1cf548d31f60ae783138e5fd42013f6271049d7df7afadef96c" +checksum = "1a87aa2bb7d2af34197c04845522473242e1aa17c12f4935d5856491a7fb8c97" dependencies = [ "cesu8", + "cfg-if", "combine", "jni-sys", "log", "thiserror", "walkdir", + "windows-sys 0.45.0", ] [[package]] @@ -1477,6 +1553,17 @@ dependencies = [ "serde_json", ] +[[package]] +name = "keyboard-types" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b750dcadc39a09dbadd74e118f6dd6598df77fa01df0cfcdc52c28dece74528a" +dependencies = [ + "bitflags 2.6.0", + "serde", + "unicode-segmentation", +] + [[package]] name = "kuchikiki" version = "0.8.2" @@ -1496,12 +1583,46 @@ version = "1.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "bbd2bcb4c963f2ddae06a2efc7e9f3591312473c50c6685e1f298068316e66fe" +[[package]] +name = "libappindicator" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "03589b9607c868cc7ae54c0b2a22c8dc03dd41692d48f2d7df73615c6a95dc0a" +dependencies = [ + "glib", + "gtk", + "gtk-sys", + "libappindicator-sys", + "log", +] + +[[package]] +name = "libappindicator-sys" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6e9ec52138abedcc58dc17a7c6c0c00a2bdb4f3427c7f63fa97fd0d859155caf" +dependencies = [ + "gtk-sys", + "libloading", + "once_cell", +] + [[package]] name = "libc" version = "0.2.158" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d8adc4bb1803a324070e64a98ae98f38934d91957a99cfb3a43dcbc01bc56439" +[[package]] +name = "libloading" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b67380fd3b2fbe7527a606e18729d21c6f3951633d0500574c4dc22d2d638b9f" +dependencies = [ + "cfg-if", + "winapi", +] + [[package]] name = "libredox" version = "0.1.3" @@ -1510,15 +1631,8 @@ checksum = "c0ff37bd590ca25063e35af745c343cb7a0271906fb7b37e4813e8f79f00268d" dependencies = [ "bitflags 2.6.0", "libc", - "redox_syscall", ] -[[package]] -name = "linux-raw-sys" -version = "0.4.14" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "78b3ae25bc7c8c38cec158d1f2757ee79e9b3740fbc7ccf0e59e4b08d793fa89" - [[package]] name = "lock_api" version = "0.4.12" @@ -1535,21 +1649,6 @@ 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 = "mac" version = "0.1.1" @@ -1579,15 +1678,6 @@ dependencies = [ "tendril", ] -[[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 = "matches" version = "0.1.10" @@ -1619,6 +1709,12 @@ dependencies = [ "autocfg", ] +[[package]] +name = "mime" +version = "0.3.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6877bb514081ee2a7ff5ef9de3281f14a4dd4bceac4c09388074a6b5df8a139a" + [[package]] name = "miniz_oxide" version = "0.7.4" @@ -1650,16 +1746,38 @@ dependencies = [ "windows-sys 0.52.0", ] +[[package]] +name = "muda" +version = "0.15.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b8123dfd4996055ac9b15a60ad263b44b01e539007523ad7a4a533a3d93b0591" +dependencies = [ + "crossbeam-channel", + "dpi", + "gtk", + "keyboard-types", + "objc2", + "objc2-app-kit", + "objc2-foundation", + "once_cell", + "png", + "serde", + "thiserror", + "windows-sys 0.59.0", +] + [[package]] name = "ndk" -version = "0.6.0" +version = "0.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2032c77e030ddee34a6787a64166008da93f6a352b629261d0fee232b8742dd4" +checksum = "c3f42e7bbe13d351b6bead8286a43aac9534b82bd3cc43e47037f012ebfd62d4" dependencies = [ - "bitflags 1.3.2", + "bitflags 2.6.0", "jni-sys", + "log", "ndk-sys", "num_enum", + "raw-window-handle", "thiserror", ] @@ -1671,9 +1789,9 @@ checksum = "27b02d87554356db9e9a873add8782d4ea6e3e58ea071a9adb9a2e8ddb884a8b" [[package]] name = "ndk-sys" -version = "0.3.0" +version = "0.6.0+11769913" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6e5a6ae77c8ee183dcbbba6150e2e6b9f3f4196a7666c02a715a95692ec1fa97" +checksum = "ee6cda3051665f1fb8d9e08fc35c96d5a244fb1be711a03b71118828afc9a873" dependencies = [ "jni-sys", ] @@ -1690,16 +1808,6 @@ version = "0.1.14" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "72ef4a56884ca558e5ddb05a1d1e7e1bfd9a68d9ed024c21704cc98872dae1bb" -[[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-conv" version = "0.1.0" @@ -1717,23 +1825,23 @@ dependencies = [ [[package]] name = "num_enum" -version = "0.5.11" +version = "0.7.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1f646caf906c20226733ed5b1374287eb97e3c2a5c227ce668c1f2ce20ae57c9" +checksum = "4e613fc340b2220f734a8595782c551f1250e969d87d3be1ae0579e8d4065179" dependencies = [ "num_enum_derive", ] [[package]] name = "num_enum_derive" -version = "0.5.11" +version = "0.7.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dcbff9bc912032c62bf65ef1d5aea88983b420f4f839db1e9b0c281a25c9c799" +checksum = "af1844ef2428cc3e1cb900be36181049ef3d3193c63e43026cfe202983b27a56" dependencies = [ - "proc-macro-crate", + "proc-macro-crate 2.0.0", "proc-macro2", "quote", - "syn 1.0.109", + "syn 2.0.77", ] [[package]] @@ -1746,6 +1854,105 @@ dependencies = [ "objc_exception", ] +[[package]] +name = "objc-sys" +version = "0.3.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cdb91bdd390c7ce1a8607f35f3ca7151b65afc0ff5ff3b34fa350f7d7c7e4310" + +[[package]] +name = "objc2" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "46a785d4eeff09c14c487497c162e92766fbb3e4059a71840cecc03d9a50b804" +dependencies = [ + "objc-sys", + "objc2-encode", +] + +[[package]] +name = "objc2-app-kit" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e4e89ad9e3d7d297152b17d39ed92cd50ca8063a89a9fa569046d41568891eff" +dependencies = [ + "bitflags 2.6.0", + "block2", + "libc", + "objc2", + "objc2-core-data", + "objc2-core-image", + "objc2-foundation", + "objc2-quartz-core", +] + +[[package]] +name = "objc2-core-data" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "617fbf49e071c178c0b24c080767db52958f716d9eabdf0890523aeae54773ef" +dependencies = [ + "bitflags 2.6.0", + "block2", + "objc2", + "objc2-foundation", +] + +[[package]] +name = "objc2-core-image" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "55260963a527c99f1819c4f8e3b47fe04f9650694ef348ffd2227e8196d34c80" +dependencies = [ + "block2", + "objc2", + "objc2-foundation", + "objc2-metal", +] + +[[package]] +name = "objc2-encode" +version = "4.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7891e71393cd1f227313c9379a26a584ff3d7e6e7159e988851f0934c993f0f8" + +[[package]] +name = "objc2-foundation" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0ee638a5da3799329310ad4cfa62fbf045d5f56e3ef5ba4149e7452dcf89d5a8" +dependencies = [ + "bitflags 2.6.0", + "block2", + "libc", + "objc2", +] + +[[package]] +name = "objc2-metal" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dd0cba1276f6023976a406a14ffa85e1fdd19df6b0f737b063b95f6c8c7aadd6" +dependencies = [ + "bitflags 2.6.0", + "block2", + "objc2", + "objc2-foundation", +] + +[[package]] +name = "objc2-quartz-core" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e42bee7bff906b14b167da2bac5efe6b6a07e6f7c0a21a7308d40c960242dc7a" +dependencies = [ + "bitflags 2.6.0", + "block2", + "objc2", + "objc2-foundation", + "objc2-metal", +] + [[package]] name = "objc_exception" version = "0.1.2" @@ -1781,27 +1988,38 @@ checksum = "3fdb12b2476b595f9358c5161aa467c2438859caa136dec86c26fdd2efe17b92" [[package]] name = "open" -version = "3.2.0" +version = "5.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2078c0039e6a54a0c42c28faa984e115fb4c2d5bf2208f77d1961002df8576f8" +checksum = "61a877bf6abd716642a53ef1b89fb498923a4afca5c754f9050b4d081c05c4b3" dependencies = [ + "is-wsl", + "libc", "pathdiff", - "windows-sys 0.42.0", ] [[package]] -name = "overload" -version = "0.1.1" +name = "option-ext" +version = "0.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b15813163c1d831bf4a13c3610c05c0d03b39feb07f7e09fa234dac9b15aaf39" +checksum = "04744f49eae99ab78e0d5c0b603ab218f515ea8cfe5a456d7629ad883a3b6e7d" + +[[package]] +name = "os_pipe" +version = "1.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5ffd2b0a5634335b135d5728d84c5e0fd726954b87111f7506a61c502280d982" +dependencies = [ + "libc", + "windows-sys 0.59.0", +] [[package]] name = "pango" -version = "0.15.10" +version = "0.18.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "22e4045548659aee5313bde6c582b0d83a627b7904dd20dc2d9ef0895d414e4f" +checksum = "7ca27ec1eb0457ab26f3036ea52229edbdb74dee1edd29063f5b9b010e7ebee4" dependencies = [ - "bitflags 1.3.2", + "gio", "glib", "libc", "once_cell", @@ -1810,14 +2028,14 @@ dependencies = [ [[package]] name = "pango-sys" -version = "0.15.10" +version = "0.18.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d2a00081cde4661982ed91d80ef437c20eacaf6aa1a5962c0279ae194662c3aa" +checksum = "436737e391a843e5933d6d9aa102cb126d501e815b83601365a948a518555dc5" dependencies = [ "glib-sys", "gobject-sys", "libc", - "system-deps 6.2.2", + "system-deps", ] [[package]] @@ -2094,6 +2312,15 @@ dependencies = [ "toml_edit 0.19.15", ] +[[package]] +name = "proc-macro-crate" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7e8366a6159044a37876a2b9817124296703c586a5c92e2c53751fa06d8d43e8" +dependencies = [ + "toml_edit 0.20.7", +] + [[package]] name = "proc-macro-error" version = "1.0.4" @@ -2234,9 +2461,9 @@ dependencies = [ [[package]] name = "raw-window-handle" -version = "0.5.2" +version = "0.6.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f2ff9a1f06a88b01621b7ae906ef0211290d1c8a168a15542486a8f61c0833b9" +checksum = "20675572f6f24e9e76ef639bc5552774ed45f1c30e2951e1e99c59888861c539" [[package]] name = "redox_syscall" @@ -2266,41 +2493,63 @@ checksum = "4219d74c6b67a3654a9fbebc4b419e22126d13d2f3c4a07ee0cb61ff79a79619" dependencies = [ "aho-corasick", "memchr", - "regex-automata 0.4.7", - "regex-syntax 0.8.4", + "regex-automata", + "regex-syntax", ] [[package]] name = "regex-automata" -version = "0.1.10" +version = "0.4.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6c230d73fb8d8c1b9c0b3135c5142a8acee3a0558fb8db5cf1cb65f8d7862132" +checksum = "38caf58cc5ef2fed281f89292ef23f6365465ed9a41b7a7754eb4e26496c92df" 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", + "aho-corasick", + "memchr", + "regex-syntax", ] [[package]] name = "regex-syntax" -version = "0.6.29" +version = "0.8.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f162c6dd7b008981e4d40210aca20b4bd0f9b60ca9271061b07f78537722f2e1" +checksum = "7a66a03ae7c801facd77a29370b4faec201768915ac14a721ba36f20bc9c209b" [[package]] -name = "regex-syntax" -version = "0.8.4" +name = "reqwest" +version = "0.12.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7a66a03ae7c801facd77a29370b4faec201768915ac14a721ba36f20bc9c209b" +checksum = "f713147fbe92361e52392c73b8c9e48c04c6625bce969ef54dc901e58e042a7b" +dependencies = [ + "base64 0.22.1", + "bytes", + "futures-core", + "futures-util", + "http", + "http-body", + "http-body-util", + "hyper", + "hyper-util", + "ipnet", + "js-sys", + "log", + "mime", + "once_cell", + "percent-encoding", + "pin-project-lite", + "serde", + "serde_json", + "serde_urlencoded", + "sync_wrapper", + "tokio", + "tokio-util", + "tower-service", + "url", + "wasm-bindgen", + "wasm-bindgen-futures", + "wasm-streams", + "web-sys", + "windows-registry", +] [[package]] name = "rustc-demangle" @@ -2317,25 +2566,6 @@ dependencies = [ "semver", ] -[[package]] -name = "rustix" -version = "0.38.37" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8acb788b847c24f28525660c4d7758620a7210875711f79e7f663cc152726811" -dependencies = [ - "bitflags 2.6.0", - "errno", - "libc", - "linux-raw-sys", - "windows-sys 0.52.0", -] - -[[package]] -name = "rustversion" -version = "1.0.17" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "955d28af4278de8121b7ebeb796b6a45735dc01436d898801014aced2773a3d6" - [[package]] name = "ryu" version = "1.0.18" @@ -2352,10 +2582,31 @@ dependencies = [ ] [[package]] -name = "scoped-tls" -version = "1.0.1" +name = "schemars" +version = "0.8.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09c024468a378b7e36765cd36702b7a90cc3cba11654f6685c8f233408e89e92" +dependencies = [ + "dyn-clone", + "indexmap 1.9.3", + "schemars_derive", + "serde", + "serde_json", + "url", + "uuid", +] + +[[package]] +name = "schemars_derive" +version = "0.8.21" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e1cf6437eb19a8f4a6cc0f7dca544973b0b78843adbfeb3683d1a94a0024a294" +checksum = "b1eee588578aff73f856ab961cd2f79e36bc45d7ded33a7562adba4667aecc0e" +dependencies = [ + "proc-macro2", + "quote", + "serde_derive_internals", + "syn 2.0.77", +] [[package]] name = "scopeguard" @@ -2401,6 +2652,17 @@ dependencies = [ "serde_derive", ] +[[package]] +name = "serde-untagged" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2676ba99bd82f75cae5cbd2c8eda6fa0b8760f18978ea840e980dd5567b5c5b6" +dependencies = [ + "erased-serde", + "serde", + "typeid", +] + [[package]] name = "serde_derive" version = "1.0.210" @@ -2412,13 +2674,23 @@ dependencies = [ "syn 2.0.77", ] +[[package]] +name = "serde_derive_internals" +version = "0.29.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "18d26a20a969b9e3fdf2fc2d9f21eda6c40e2de84c9408bb5d3b05d499aae711" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.77", +] + [[package]] name = "serde_json" version = "1.0.128" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6ff5456707a1de34e7e37f2a6fd3d3f808c318259cbd01ab6377795054b483d8" dependencies = [ - "indexmap 2.5.0", "itoa 1.0.11", "memchr", "ryu", @@ -2445,6 +2717,18 @@ dependencies = [ "serde", ] +[[package]] +name = "serde_urlencoded" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3491c14715ca2294c4d6a88f15e84739788c1d030eed8c110436aafdaa2f3fd" +dependencies = [ + "form_urlencoded", + "itoa 1.0.11", + "ryu", + "serde", +] + [[package]] name = "serde_with" version = "3.9.0" @@ -2477,9 +2761,9 @@ dependencies = [ [[package]] name = "serialize-to-javascript" -version = "0.1.2" +version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "04f3666a07a197cdb77cdf306c32be9b7f598d7060d50cfd4d5aa04bfd92f6c5" +checksum = "c9823f2d3b6a81d98228151fdeaf848206a7855a7a042bbf9bf870449a66cafb" dependencies = [ "serde", "serde_json", @@ -2488,13 +2772,13 @@ dependencies = [ [[package]] name = "serialize-to-javascript-impl" -version = "0.1.2" +version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "772ee033c0916d670af7860b6e1ef7d658a4629a6d0b4c8c3e67f09b3765b75d" +checksum = "74064874e9f6a15f04c1f3cb627902d0e6b410abbf36668afa873c61889f1763" dependencies = [ "proc-macro2", "quote", - "syn 2.0.77", + "syn 1.0.109", ] [[package]] @@ -2519,12 +2803,13 @@ dependencies = [ ] [[package]] -name = "sharded-slab" -version = "0.1.7" +name = "shared_child" +version = "1.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f40ca3c46823713e0d4209592e8d6e826aa57e928f09752619fc696c499637f6" +checksum = "09fa9338aed9a1df411814a5b2252f7cd206c55ae9bf2fa763f8de84603aa60c" dependencies = [ - "lazy_static", + "libc", + "windows-sys 0.59.0", ] [[package]] @@ -2571,31 +2856,51 @@ dependencies = [ ] [[package]] -name = "soup2" -version = "0.2.1" +name = "softbuffer" +version = "0.4.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b2b4d76501d8ba387cf0fefbe055c3e0a59891d09f0f995ae4e4b16f6b60f3c0" +checksum = "18051cdd562e792cad055119e0cdb2cfc137e44e3987532e0f9659a77931bb08" dependencies = [ - "bitflags 1.3.2", + "bytemuck", + "cfg_aliases", + "core-graphics", + "foreign-types", + "js-sys", + "log", + "objc2", + "objc2-foundation", + "objc2-quartz-core", + "raw-window-handle", + "redox_syscall", + "wasm-bindgen", + "web-sys", + "windows-sys 0.59.0", +] + +[[package]] +name = "soup3" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "471f924a40f31251afc77450e781cb26d55c0b650842efafc9c6cbd2f7cc4f9f" +dependencies = [ + "futures-channel", "gio", "glib", "libc", - "once_cell", - "soup2-sys", + "soup3-sys", ] [[package]] -name = "soup2-sys" -version = "0.2.0" +name = "soup3-sys" +version = "0.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "009ef427103fcb17f802871647a7fa6c60cbb654b4c4e4c0ac60a31c5f6dc9cf" +checksum = "7ebe8950a680a12f24f15ebe1bf70db7af98ad242d9db43596ad3108aab86c27" dependencies = [ - "bitflags 1.3.2", "gio-sys", "glib-sys", "gobject-sys", "libc", - "system-deps 5.0.0", + "system-deps", ] [[package]] @@ -2604,15 +2909,6 @@ version = "1.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a8f112729512f8e442d81f95a8a7ddf2b7c6b8a1a6f509a95864142b30cab2d3" -[[package]] -name = "state" -version = "0.5.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dbe866e1e51e8260c9eed836a042a5e7f6726bb2b411dffeaa712e19c388f23b" -dependencies = [ - "loom", -] - [[package]] name = "string_cache" version = "0.8.7" @@ -2662,6 +2958,17 @@ version = "2.6.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "13c2bddecc57b384dee18652358fb23172facb8a2c51ccc10d74c157bdea3292" +[[package]] +name = "swift-rs" +version = "1.0.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4057c98e2e852d51fdcfca832aac7b571f6b351ad159f9eda5db1655f8d0c4d7" +dependencies = [ + "base64 0.21.7", + "serde", + "serde_json", +] + [[package]] name = "syn" version = "1.0.109" @@ -2685,16 +2992,12 @@ dependencies = [ ] [[package]] -name = "system-deps" -version = "5.0.0" +name = "sync_wrapper" +version = "1.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "18db855554db7bd0e73e06cf7ba3df39f97812cb11d3f75e71c39bf45171797e" +checksum = "a7065abeca94b6a8a577f9bd45aa0867a2238b74e8eb67cf10d492bc39351394" dependencies = [ - "cfg-expr 0.9.1", - "heck 0.3.3", - "pkg-config", - "toml 0.5.11", - "version-compare 0.0.11", + "futures-core", ] [[package]] @@ -2703,37 +3006,30 @@ version = "6.2.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a3e535eb8dded36d55ec13eddacd30dec501792ff23a0b1682c38601b8cf2349" dependencies = [ - "cfg-expr 0.15.8", + "cfg-expr", "heck 0.5.0", "pkg-config", "toml 0.8.19", - "version-compare 0.2.0", + "version-compare", ] [[package]] name = "tao" -version = "0.16.10" +version = "0.30.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "48d298c441a1da46e28e8ad8ec205aab7fd8cd71b9d10e05454224eef422e1ae" +checksum = "06e48d7c56b3f7425d061886e8ce3b6acfab1993682ed70bef50fd133d721ee6" dependencies = [ - "bitflags 1.3.2", - "cairo-rs", - "cc", + "bitflags 2.6.0", "cocoa", "core-foundation", "core-graphics", "crossbeam-channel", "dispatch", - "gdk", - "gdk-pixbuf", - "gdk-sys", + "dlopen2", + "dpi", "gdkwayland-sys", "gdkx11-sys", - "gio", - "glib", - "glib-sys", "gtk", - "image", "instant", "jni", "lazy_static", @@ -2745,15 +3041,14 @@ dependencies = [ "objc", "once_cell", "parking_lot", - "png", "raw-window-handle", "scopeguard", - "serde", "tao-macros", "unicode-segmentation", - "uuid", - "windows 0.39.0", - "windows-implement", + "url", + "windows", + "windows-core 0.58.0", + "windows-version", "x11-dl", ] @@ -2768,17 +3063,6 @@ dependencies = [ "syn 2.0.77", ] -[[package]] -name = "tar" -version = "0.4.41" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cb797dad5fb5b76fcf519e702f4a589483b5ef06567f160c392832c1f5e44909" -dependencies = [ - "filetime", - "libc", - "xattr", -] - [[package]] name = "target-lexicon" version = "0.12.16" @@ -2787,81 +3071,83 @@ checksum = "61c41af27dd6d1e27b1b16b489db798443478cef1f06a660c96db617ba5de3b1" [[package]] name = "tauri" -version = "1.8.0" +version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "570a20223602ad990a30a048f2fdb957ae3e38de3ca9582e04cc09d01e8ccfad" +checksum = "3c9c08beea86d5095b6f5fb1c788fe8759b23c3f71927c66a69e725a91d089cd" dependencies = [ "anyhow", - "cocoa", - "dirs-next", + "bytes", + "dirs", "dunce", "embed_plist", - "encoding_rs", - "flate2", "futures-util", "getrandom 0.2.15", - "glib", "glob", "gtk", "heck 0.5.0", "http", - "ignore", + "jni", + "libc", "log", - "objc", - "once_cell", - "open", + "mime", + "muda", + "objc2", + "objc2-app-kit", + "objc2-foundation", "percent-encoding", "plist", - "rand 0.8.5", "raw-window-handle", - "regex", - "semver", + "reqwest", "serde", "serde_json", "serde_repr", "serialize-to-javascript", - "state", - "tar", + "swift-rs", + "tauri-build", "tauri-macros", "tauri-runtime", "tauri-runtime-wry", "tauri-utils", - "tempfile", "thiserror", "tokio", + "tray-icon", "url", - "uuid", + "urlpattern", "webkit2gtk", "webview2-com", - "windows 0.39.0", + "window-vibrancy", + "windows", ] [[package]] name = "tauri-build" -version = "1.5.5" +version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "586f3e677f940c8bb4f70c52eda05dc59b79e61543f1182de83516810bb8e35d" +checksum = "93bb649a284aec2ab43e8df6831b8c8060d231ec8ddf05bf021d58cb67570e1f" dependencies = [ "anyhow", "cargo_toml", - "dirs-next", + "dirs", + "glob", "heck 0.5.0", "json-patch", + "schemars", "semver", "serde", "serde_json", "tauri-utils", "tauri-winres", + "toml 0.8.19", "walkdir", ] [[package]] name = "tauri-codegen" -version = "1.4.5" +version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "93a9e3f5cebf779a63bf24903e714ec91196c307d8249a0008b882424328bcda" +checksum = "a4511912612ba0da11aeb300e18e18b2c7067fd14aa886eac46bdcc43b4fa3ee" dependencies = [ - "base64 0.21.7", + "base64 0.22.1", "brotli", "ico", "json-patch", @@ -2869,84 +3155,148 @@ dependencies = [ "png", "proc-macro2", "quote", - "regex", "semver", "serde", "serde_json", "sha2", + "syn 2.0.77", "tauri-utils", "thiserror", "time", + "url", "uuid", "walkdir", ] [[package]] name = "tauri-macros" -version = "1.4.6" +version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d1d0e989f54fe06c5ef0875c5e19cf96453d099a0a774d5192ab47e80471cdab" +checksum = "62ee976578a14b779996d7b6879d7e625c8ce674bc87e223953664f37def2eef" dependencies = [ "heck 0.5.0", "proc-macro2", "quote", - "syn 1.0.109", + "syn 2.0.77", "tauri-codegen", "tauri-utils", ] +[[package]] +name = "tauri-plugin" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "774d084450b7ec8e445ad119079307f935b7bf3d736da139a8664eb1d4909aa5" +dependencies = [ + "anyhow", + "glob", + "plist", + "schemars", + "serde", + "serde_json", + "tauri-utils", + "toml 0.8.19", + "walkdir", +] + +[[package]] +name = "tauri-plugin-fs" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6ba59d7880ffdc24f72350b48bec374482803e167b7275655f6bd8426b7fc9c3" +dependencies = [ + "anyhow", + "dunce", + "glob", + "percent-encoding", + "schemars", + "serde", + "serde_json", + "serde_repr", + "tauri", + "tauri-plugin", + "thiserror", + "url", + "uuid", +] + +[[package]] +name = "tauri-plugin-shell" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2929bb35edb7255949e0cbcb2285ff6b02371bf826ad03471077b6b3bf4e6d60" +dependencies = [ + "encoding_rs", + "log", + "open", + "os_pipe", + "regex", + "schemars", + "serde", + "serde_json", + "shared_child", + "tauri", + "tauri-plugin", + "thiserror", + "tokio", +] + [[package]] name = "tauri-runtime" -version = "0.14.5" +version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f33fda7d213e239077fad52e96c6b734cecedb30c2382118b64f94cb5103ff3a" +checksum = "2570e1f33f332a2d2d9967ebb3903bc4e1f92b9c47e4d1b302c10ea4153fcdbb" dependencies = [ + "dpi", "gtk", "http", - "http-range", - "rand 0.8.5", + "jni", "raw-window-handle", "serde", "serde_json", "tauri-utils", "thiserror", "url", - "uuid", - "webview2-com", - "windows 0.39.0", + "windows", ] [[package]] name = "tauri-runtime-wry" -version = "0.14.10" +version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "18c447dcd9b0f09c7dc4b752cc33e72788805bfd761fbda5692d30c48289efec" +checksum = "8147d8f9ed418d83a90af3d64fbdca5e0e924ae28e5351da88f9568169db8665" dependencies = [ - "cocoa", "gtk", + "http", + "jni", + "log", + "objc2", + "objc2-app-kit", + "objc2-foundation", "percent-encoding", - "rand 0.8.5", "raw-window-handle", + "softbuffer", + "tao", "tauri-runtime", "tauri-utils", - "uuid", + "url", "webkit2gtk", "webview2-com", - "windows 0.39.0", + "windows", "wry", ] [[package]] name = "tauri-utils" -version = "1.6.1" +version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "83a0c939e88d82903a0a7dfb28388b12a3c03504d6bd6086550edaa3b6d8beaa" +checksum = "f87856e9d7fa91fd710362f3c73fccbf6bfd036934908791e65bd803d54dc8a8" dependencies = [ "brotli", + "cargo_metadata", "ctor", "dunce", "glob", - "heck 0.5.0", "html5ever", "infer", "json-patch", @@ -2956,14 +3306,20 @@ dependencies = [ "phf 0.11.2", "proc-macro2", "quote", + "regex", + "schemars", "semver", "serde", + "serde-untagged", "serde_json", "serde_with", + "swift-rs", "thiserror", + "toml 0.8.19", "url", + "urlpattern", + "uuid", "walkdir", - "windows-version", ] [[package]] @@ -2976,19 +3332,6 @@ dependencies = [ "toml 0.7.8", ] -[[package]] -name = "tempfile" -version = "3.12.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "04cbcdd0c794ebb0d4cf35e88edd2f7d2c4c3e9a5a6dab322839b321c6a87a64" -dependencies = [ - "cfg-if", - "fastrand", - "once_cell", - "rustix", - "windows-sys 0.59.0", -] - [[package]] name = "tendril" version = "0.4.3" @@ -3026,16 +3369,6 @@ dependencies = [ "syn 2.0.77", ] -[[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 = "time" version = "0.3.36" @@ -3136,15 +3469,6 @@ dependencies = [ "tokio", ] -[[package]] -name = "toml" -version = "0.5.11" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f4f7f0dd8d50a853a531c426359045b1998f04219d88799810762cd4ad314234" -dependencies = [ - "serde", -] - [[package]] name = "toml" version = "0.7.8" @@ -3191,6 +3515,17 @@ dependencies = [ "winnow 0.5.40", ] +[[package]] +name = "toml_edit" +version = "0.20.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "70f427fce4d84c72b5b732388bf4a9f4531b53f74e2887e3ecb2481f68f66d81" +dependencies = [ + "indexmap 2.5.0", + "toml_datetime", + "winnow 0.5.40", +] + [[package]] name = "toml_edit" version = "0.22.21" @@ -3204,6 +3539,12 @@ dependencies = [ "winnow 0.6.18", ] +[[package]] +name = "tower-service" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8df9b6e13f2d32c91b9bd719c00d1958837bc7dec474d94952798cc8e69eeec3" + [[package]] name = "tracing" version = "0.1.40" @@ -3211,21 +3552,9 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c3523ab5a71916ccf420eebdf5521fcef02141234bbc0b8a49f2fdc4544364ef" dependencies = [ "pin-project-lite", - "tracing-attributes", "tracing-core", ] -[[package]] -name = "tracing-attributes" -version = "0.1.27" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "34704c8d6ebcbc939824180af020566b01a7c01f80641264eba0999f6c2b6be7" -dependencies = [ - "proc-macro2", - "quote", - "syn 2.0.77", -] - [[package]] name = "tracing-core" version = "0.1.32" @@ -3233,37 +3562,40 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c06d3da6113f116aaee68e4d601191614c9053067f9ab7f6edbcb161237daa54" dependencies = [ "once_cell", - "valuable", ] [[package]] -name = "tracing-log" -version = "0.2.0" +name = "tray-icon" +version = "0.19.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ee855f1f400bd0e5c02d150ae5de3840039a3f54b025156404e34c23c03f47c3" +checksum = "533fc2d4105e0e3d96ce1c71f2d308c9fbbe2ef9c587cab63dd627ab5bde218f" dependencies = [ - "log", + "core-graphics", + "crossbeam-channel", + "dirs", + "libappindicator", + "muda", + "objc2", + "objc2-app-kit", + "objc2-foundation", "once_cell", - "tracing-core", + "png", + "serde", + "thiserror", + "windows-sys 0.59.0", ] [[package]] -name = "tracing-subscriber" -version = "0.3.18" +name = "try-lock" +version = "0.2.5" 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", -] +checksum = "e421abadd41a4225275504ea4d6566923418b7f05506fbc9c0fe86ba7396114b" + +[[package]] +name = "typeid" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0e13db2e0ccd5e14a544e8a246ba2312cd25223f616442d7f2cb0e3db614236e" [[package]] name = "typenum" @@ -3271,6 +3603,47 @@ version = "1.17.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825" +[[package]] +name = "unic-char-property" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a8c57a407d9b6fa02b4795eb81c5b6652060a15a7903ea981f3d723e6c0be221" +dependencies = [ + "unic-char-range", +] + +[[package]] +name = "unic-char-range" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0398022d5f700414f6b899e10b8348231abf9173fa93144cbc1a43b9793c1fbc" + +[[package]] +name = "unic-common" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "80d7ff825a6a654ee85a63e80f92f054f904f21e7d12da4e22f9834a4aaa35bc" + +[[package]] +name = "unic-ucd-ident" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e230a37c0381caa9219d67cf063aa3a375ffed5bf541a452db16e744bdab6987" +dependencies = [ + "unic-char-property", + "unic-char-range", + "unic-ucd-version", +] + +[[package]] +name = "unic-ucd-version" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "96bd2f2237fe450fcd0a1d2f5f4e91711124f7857ba2e964247776ebeeb7b0c4" +dependencies = [ + "unic-common", +] + [[package]] name = "unicode-bidi" version = "0.3.15" @@ -3316,6 +3689,18 @@ dependencies = [ "serde", ] +[[package]] +name = "urlpattern" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "70acd30e3aa1450bc2eece896ce2ad0d178e9c079493819301573dae3c37ba6d" +dependencies = [ + "regex", + "serde", + "unic-ucd-ident", + "url", +] + [[package]] name = "utf-8" version = "0.7.6" @@ -3329,20 +3714,9 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "81dfa00651efa65069b0b6b651f4aaa31ba9e3c3ce0137aaad053604ee7e0314" dependencies = [ "getrandom 0.2.15", + "serde", ] -[[package]] -name = "valuable" -version = "0.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "830b7e5d4d90034032940e4ace0d9a9a057e7a45cd94e6c007832e39edb82f6d" - -[[package]] -name = "version-compare" -version = "0.0.11" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1c18c859eead79d8b95d09e4678566e8d70105c4e7b251f707a03df32442661b" - [[package]] name = "version-compare" version = "0.2.0" @@ -3385,6 +3759,15 @@ dependencies = [ "winapi-util", ] +[[package]] +name = "want" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bfa7760aed19e106de2c7c0b581b509f2f25d3dacaf737cb82ac61bc6d760b0e" +dependencies = [ + "try-lock", +] + [[package]] name = "wasi" version = "0.9.0+wasi-snapshot-preview1" @@ -3429,6 +3812,18 @@ dependencies = [ "wasm-bindgen-shared", ] +[[package]] +name = "wasm-bindgen-futures" +version = "0.4.43" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "61e9300f63a621e96ed275155c108eb6f843b6a26d053f122ab69724559dc8ed" +dependencies = [ + "cfg-if", + "js-sys", + "wasm-bindgen", + "web-sys", +] + [[package]] name = "wasm-bindgen-macro" version = "0.2.93" @@ -3458,6 +3853,19 @@ version = "0.2.93" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c62a0a307cb4a311d3a07867860911ca130c3494e8c2719593806c08bc5d0484" +[[package]] +name = "wasm-streams" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4e072d4e72f700fb3443d8fe94a39315df013eef1104903cdb0a2abd322bbecd" +dependencies = [ + "futures-util", + "js-sys", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", +] + [[package]] name = "web-sys" version = "0.3.70" @@ -3470,9 +3878,9 @@ dependencies = [ [[package]] name = "webkit2gtk" -version = "0.18.2" +version = "2.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b8f859735e4a452aeb28c6c56a852967a8a76c8eb1cc32dbf931ad28a13d6370" +checksum = "76b1bc1e54c581da1e9f179d0b38512ba358fb1af2d634a1affe42e37172361a" dependencies = [ "bitflags 1.3.2", "cairo-rs", @@ -3488,20 +3896,18 @@ dependencies = [ "javascriptcore-rs", "libc", "once_cell", - "soup2", + "soup3", "webkit2gtk-sys", ] [[package]] name = "webkit2gtk-sys" -version = "0.18.0" +version = "2.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4d76ca6ecc47aeba01ec61e480139dda143796abcae6f83bcddf50d6b5b1dcf3" +checksum = "62daa38afc514d1f8f12b8693d30d5993ff77ced33ce30cd04deebc267a6d57c" dependencies = [ - "atk-sys", "bitflags 1.3.2", "cairo-sys-rs", - "gdk-pixbuf-sys", "gdk-sys", "gio-sys", "glib-sys", @@ -3509,48 +3915,45 @@ dependencies = [ "gtk-sys", "javascriptcore-rs-sys", "libc", - "pango-sys", "pkg-config", - "soup2-sys", - "system-deps 6.2.2", + "soup3-sys", + "system-deps", ] [[package]] name = "webview2-com" -version = "0.19.1" +version = "0.33.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b4a769c9f1a64a8734bde70caafac2b96cada12cd4aefa49196b3a386b8b4178" +checksum = "6f61ff3d9d0ee4efcb461b14eb3acfda2702d10dc329f339303fc3e57215ae2c" dependencies = [ "webview2-com-macros", "webview2-com-sys", - "windows 0.39.0", + "windows", + "windows-core 0.58.0", "windows-implement", + "windows-interface", ] [[package]] name = "webview2-com-macros" -version = "0.6.0" +version = "0.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "eaebe196c01691db62e9e4ca52c5ef1e4fd837dcae27dae3ada599b5a8fd05ac" +checksum = "1d228f15bba3b9d56dde8bddbee66fa24545bd17b48d5128ccf4a8742b18e431" dependencies = [ "proc-macro2", "quote", - "syn 1.0.109", + "syn 2.0.77", ] [[package]] name = "webview2-com-sys" -version = "0.19.0" +version = "0.33.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "aac48ef20ddf657755fdcda8dfed2a7b4fc7e4581acce6fe9b88c3d64f29dee7" +checksum = "a3a3e2eeb58f82361c93f9777014668eb3d07e7d174ee4c819575a9208011886" dependencies = [ - "regex", - "serde", - "serde_json", "thiserror", - "windows 0.39.0", - "windows-bindgen", - "windows-metadata", + "windows", + "windows-core 0.58.0", ] [[package]] @@ -3596,76 +3999,110 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" [[package]] -name = "windows" -version = "0.39.0" +name = "window-vibrancy" +version = "0.5.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f1c4bd0a50ac6020f65184721f758dba47bb9fbc2133df715ec74a237b26794a" +checksum = "3ea403deff7b51fff19e261330f71608ff2cdef5721d72b64180bb95be7c4150" dependencies = [ - "windows-implement", - "windows_aarch64_msvc 0.39.0", - "windows_i686_gnu 0.39.0", - "windows_i686_msvc 0.39.0", - "windows_x86_64_gnu 0.39.0", - "windows_x86_64_msvc 0.39.0", + "objc2", + "objc2-app-kit", + "objc2-foundation", + "raw-window-handle", + "windows-sys 0.59.0", + "windows-version", ] [[package]] name = "windows" -version = "0.48.0" +version = "0.58.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e686886bc078bc1b0b600cac0147aadb815089b6e4da64016cbd754b6342700f" +checksum = "dd04d41d93c4992d421894c18c8b43496aa748dd4c081bac0dc93eb0489272b6" dependencies = [ - "windows-targets 0.48.5", + "windows-core 0.58.0", + "windows-targets 0.52.6", ] [[package]] -name = "windows-bindgen" -version = "0.39.0" +name = "windows-core" +version = "0.52.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "68003dbd0e38abc0fb85b939240f4bce37c43a5981d3df37ccbaaa981b47cb41" +checksum = "33ab640c8d7e35bf8ba19b884ba838ceb4fba93a4e8c65a9059d08afcfc683d9" dependencies = [ - "windows-metadata", - "windows-tokens", + "windows-targets 0.52.6", ] [[package]] name = "windows-core" -version = "0.52.0" +version = "0.58.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "33ab640c8d7e35bf8ba19b884ba838ceb4fba93a4e8c65a9059d08afcfc683d9" +checksum = "6ba6d44ec8c2591c134257ce647b7ea6b20335bf6379a27dac5f1641fcf59f99" dependencies = [ + "windows-implement", + "windows-interface", + "windows-result", + "windows-strings", "windows-targets 0.52.6", ] [[package]] name = "windows-implement" -version = "0.39.0" +version = "0.58.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ba01f98f509cb5dc05f4e5fc95e535f78260f15fea8fe1a8abdd08f774f1cee7" +checksum = "2bbd5b46c938e506ecbce286b6628a02171d56153ba733b6c741fc627ec9579b" dependencies = [ - "syn 1.0.109", - "windows-tokens", + "proc-macro2", + "quote", + "syn 2.0.77", +] + +[[package]] +name = "windows-interface" +version = "0.58.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "053c4c462dc91d3b1504c6fe5a726dd15e216ba718e84a0e46a88fbe5ded3515" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.77", +] + +[[package]] +name = "windows-registry" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e400001bb720a623c1c69032f8e3e4cf09984deec740f007dd2b03ec864804b0" +dependencies = [ + "windows-result", + "windows-strings", + "windows-targets 0.52.6", ] [[package]] -name = "windows-metadata" -version = "0.39.0" +name = "windows-result" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d1043d8214f791817bab27572aaa8af63732e11bf84aa21a45a78d6c317ae0e" +dependencies = [ + "windows-targets 0.52.6", +] + +[[package]] +name = "windows-strings" +version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9ee5e275231f07c6e240d14f34e1b635bf1faa1c76c57cfd59a5cdb9848e4278" +checksum = "4cd9b125c486025df0eabcb585e62173c6c9eddcec5d117d3b6e8c30e2ee4d10" +dependencies = [ + "windows-result", + "windows-targets 0.52.6", +] [[package]] name = "windows-sys" -version = "0.42.0" +version = "0.45.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5a3e1820f08b8513f676f7ab6c1f99ff312fb97b553d30ff4dd86f9f15728aa7" +checksum = "75283be5efb2831d37ea142365f009c02ec203cd29a3ebecbc093d52315b66d0" dependencies = [ - "windows_aarch64_gnullvm 0.42.2", - "windows_aarch64_msvc 0.42.2", - "windows_i686_gnu 0.42.2", - "windows_i686_msvc 0.42.2", - "windows_x86_64_gnu 0.42.2", - "windows_x86_64_gnullvm 0.42.2", - "windows_x86_64_msvc 0.42.2", + "windows-targets 0.42.2", ] [[package]] @@ -3695,6 +4132,21 @@ dependencies = [ "windows-targets 0.52.6", ] +[[package]] +name = "windows-targets" +version = "0.42.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e5180c00cd44c9b1c88adb3693291f1cd93605ded80c250a75d472756b4d071" +dependencies = [ + "windows_aarch64_gnullvm 0.42.2", + "windows_aarch64_msvc 0.42.2", + "windows_i686_gnu 0.42.2", + "windows_i686_msvc 0.42.2", + "windows_x86_64_gnu 0.42.2", + "windows_x86_64_gnullvm 0.42.2", + "windows_x86_64_msvc 0.42.2", +] + [[package]] name = "windows-targets" version = "0.48.5" @@ -3726,12 +4178,6 @@ dependencies = [ "windows_x86_64_msvc 0.52.6", ] -[[package]] -name = "windows-tokens" -version = "0.39.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f838de2fe15fe6bac988e74b798f26499a8b21a9d97edec321e79b28d1d7f597" - [[package]] name = "windows-version" version = "0.1.1" @@ -3759,12 +4205,6 @@ version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "32a4622180e7a0ec044bb555404c800bc9fd9ec262ec147edd5989ccd0c02cd3" -[[package]] -name = "windows_aarch64_msvc" -version = "0.39.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ec7711666096bd4096ffa835238905bb33fb87267910e154b18b44eaabb340f2" - [[package]] name = "windows_aarch64_msvc" version = "0.42.2" @@ -3783,12 +4223,6 @@ version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "09ec2a7bb152e2252b53fa7803150007879548bc709c039df7627cabbd05d469" -[[package]] -name = "windows_i686_gnu" -version = "0.39.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "763fc57100a5f7042e3057e7e8d9bdd7860d330070251a73d003563a3bb49e1b" - [[package]] name = "windows_i686_gnu" version = "0.42.2" @@ -3813,12 +4247,6 @@ version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0eee52d38c090b3caa76c563b86c3a4bd71ef1a819287c19d586d7334ae8ed66" -[[package]] -name = "windows_i686_msvc" -version = "0.39.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7bc7cbfe58828921e10a9f446fcaaf649204dcfe6c1ddd712c5eebae6bda1106" - [[package]] name = "windows_i686_msvc" version = "0.42.2" @@ -3837,12 +4265,6 @@ version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "240948bc05c5e7c6dabba28bf89d89ffce3e303022809e73deaefe4f6ec56c66" -[[package]] -name = "windows_x86_64_gnu" -version = "0.39.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6868c165637d653ae1e8dc4d82c25d4f97dd6605eaa8d784b5c6e0ab2a252b65" - [[package]] name = "windows_x86_64_gnu" version = "0.42.2" @@ -3879,12 +4301,6 @@ version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "24d5b23dc417412679681396f2b49f3de8c1473deb516bd34410872eff51ed0d" -[[package]] -name = "windows_x86_64_msvc" -version = "0.39.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5e4d40883ae9cae962787ca76ba76390ffa29214667a111db9e0a1ad8377e809" - [[package]] name = "windows_x86_64_msvc" version = "0.42.2" @@ -3933,40 +4349,42 @@ dependencies = [ [[package]] name = "wry" -version = "0.24.11" +version = "0.44.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c55c80b12287eb1ff7c365fc2f7a5037cb6181bd44c9fce81c8d1cf7605ffad6" +checksum = "440600584cfbd8b0d28eace95c1f2c253db05dae43780b79380aa1e868f04c73" dependencies = [ - "base64 0.13.1", + "base64 0.22.1", "block", "cocoa", "core-graphics", "crossbeam-channel", + "dpi", "dunce", - "gdk", - "gio", - "glib", + "gdkx11", "gtk", "html5ever", "http", + "javascriptcore-rs", + "jni", "kuchikiki", "libc", - "log", + "ndk", "objc", "objc_id", "once_cell", - "serde", - "serde_json", + "percent-encoding", + "raw-window-handle", "sha2", - "soup2", - "tao", + "soup3", + "tao-macros", "thiserror", - "url", "webkit2gtk", "webkit2gtk-sys", "webview2-com", - "windows 0.39.0", - "windows-implement", + "windows", + "windows-core 0.58.0", + "windows-version", + "x11-dl", ] [[package]] @@ -3990,17 +4408,6 @@ dependencies = [ "pkg-config", ] -[[package]] -name = "xattr" -version = "1.3.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8da84f1a25939b27f6820d92aed108f83ff920fdf11a7b19366c27c4cda81d4f" -dependencies = [ - "libc", - "linux-raw-sys", - "rustix", -] - [[package]] name = "zerocopy" version = "0.7.35" diff --git a/src-tauri/Cargo.toml b/src-tauri/Cargo.toml index 4713441..4f240d0 100644 --- a/src-tauri/Cargo.toml +++ b/src-tauri/Cargo.toml @@ -5,23 +5,20 @@ description = "A Tauri App" authors = ["you"] edition = "2021" +[lib] +name = "app_lib" +crate-type = ["staticlib", "cdylib", "rlib"] + # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html [build-dependencies] -tauri-build = { version = "1", features = [] } +tauri-build = { version = "2.0.0-rc", features = [] } [dependencies] -tauri = { version = "1", features = [ - "window-unminimize", - "window-unmaximize", - "window-show", - "window-minimize", - "window-hide", - "window-close", - "window-start-dragging", - "window-maximize", - "shell-open", -] } +tauri = { version = "2.0.0-rc", features = [] } +tauri-plugin-fs = "2.0.0-rc" +tauri-plugin-shell = "2.0.0-rc" + serde = { version = "1", features = ["derive"] } serde_json = "1" futures = "0.3" diff --git a/src-tauri/capabilities/migrated.json b/src-tauri/capabilities/migrated.json new file mode 100644 index 0000000..476f51b --- /dev/null +++ b/src-tauri/capabilities/migrated.json @@ -0,0 +1,41 @@ +{ + "identifier": "migrated", + "description": "permissions that were migrated from v1", + "local": true, + "windows": [ + "main" + ], + "permissions": [ + "core:default", + "fs:allow-read-file", + "fs:allow-write-file", + "fs:allow-read-dir", + "fs:allow-copy-file", + "fs:allow-mkdir", + "fs:allow-remove", + "fs:allow-remove", + "fs:allow-rename", + "fs:allow-exists", + { + "identifier": "fs:scope", + "allow": [ + "$APPCONFIG", + "$APPCONFIG/*", + "$APPCONFIG/configs", + "$APPCONFIG/configs/*" + ] + }, + "core:window:allow-maximize", + "core:window:allow-unmaximize", + "core:window:allow-minimize", + "core:window:allow-unminimize", + "core:window:allow-show", + "core:window:allow-hide", + "core:window:allow-close", + "core:window:allow-start-dragging", + "shell:allow-open", + "fs:default", + "shell:default", + "fs:default" + ] +} \ No newline at end of file diff --git a/src-tauri/src/lib.rs b/src-tauri/src/lib.rs new file mode 100644 index 0000000..9316fd2 --- /dev/null +++ b/src-tauri/src/lib.rs @@ -0,0 +1,154 @@ +// Prevents additional console window on Windows in release, DO NOT REMOVE!! +#![cfg_attr(not(debug_assertions), windows_subsystem = "windows")] + +use std::{ + any::Any, + collections::HashMap, + time::{Instant, SystemTime}, +}; + +use chrono::{DateTime, NaiveDateTime, Utc}; +use serde::{Deserialize, Serialize}; +use tauri::Manager; +use tokio_postgres::{ + types::{FromSql, Timestamp, ToSql, Type}, + Error, NoTls, +}; + +// Learn more about Tauri commands at https://tauri.app/v1/guides/features/command +#[tauri::command] +fn greet(name: &str) -> HashMap { + format!("Hello, {}! You've been greeted from Rust!", name); + let mut map: HashMap = HashMap::new(); + map.insert("name".to_string(), "John".to_string()); + map.insert("age".to_string(), "20".to_string()); + return map; +} + +#[derive(Debug, Serialize, Deserialize, Clone)] +#[serde(untagged)] +enum SqlValue { + String(Option), + Int(Option), + Float32(Option), + Float64(Option), + Bool(Option), + Timestamp(Option), + Null, +} + +#[derive(Debug, Serialize, Deserialize, Clone)] +struct ColInfo { + name: String, + col_type: String, + order_idx: i32, +} + +#[derive(Debug, Serialize, Deserialize, Clone)] +struct ExecuteQueryResult { + columns: Vec, + rows: Vec>, +} + +#[tauri::command] +async fn pg_execute_query( + host: String, + port: u16, + username: String, + password: String, + database: String, + query: String, + params: Vec, +) -> Result { + let connection_string = format!( + "postgres://{}:{}@{}:{}/{}", + username, password, host, port, database + ); + + // Connect to the database. + let (client, connection) = tokio_postgres::connect(&connection_string, NoTls) + .await + .map_err(|e| e.to_string())?; + + tokio::spawn(async move { + if let Err(e) = connection.await { + eprintln!("connection error: {}", e); + } + }); + + let params: Vec<&(dyn ToSql + Sync)> = params + .iter() + .map(|p| match p { + SqlValue::String(s) => s as &(dyn ToSql + Sync), + + SqlValue::Int(i) => i as &(dyn ToSql + Sync), + SqlValue::Float32(f) => f as &(dyn ToSql + Sync), + SqlValue::Float64(f) => f as &(dyn ToSql + Sync), + SqlValue::Bool(b) => b as &(dyn ToSql + Sync), + SqlValue::Timestamp(t) => t as &(dyn ToSql + Sync), + + SqlValue::Null => &"", + }) + .collect(); + + // Execute the query and fetch the results. + let rows = client + .query(&query, ¶ms) + .await + .map_err(|e| e.to_string())?; + + let mut result: ExecuteQueryResult = ExecuteQueryResult { + columns: Vec::new(), + rows: Vec::new(), + }; + + if rows.is_empty() { + return Ok(result); + } + + for (i, column) in rows[0].columns().iter().enumerate() { + result.columns.push(ColInfo { + name: column.name().to_string(), + col_type: column.type_().name().to_string(), + order_idx: i as i32, + }); + } + + for row in rows { + let mut row_map: HashMap = HashMap::new(); + for (i, column) in row.columns().iter().enumerate() { + let value = match column.type_() { + &Type::VARCHAR => SqlValue::String(row.get::<_, Option>(i)), + // &Type::TIMESTAMP => SqlValue::String(row.get::<_, Option>(i)), + &Type::TEXT => SqlValue::String(row.get::<_, Option>(i)), + &Type::NAME => SqlValue::String(row.get::<_, Option>(i)), + &Type::INT4 => SqlValue::Int(row.get::<_, Option>(i)), + &Type::BOOL => SqlValue::Bool(row.get::<_, Option>(i)), + &Type::FLOAT4 => SqlValue::Float32(row.get::<_, Option>(i)), + &Type::FLOAT8 => SqlValue::Float64(row.get::<_, Option>(i)), + &Type::TIMESTAMP => SqlValue::Timestamp(row.get::<_, Option>(i)), + + _ => { + println!(">>> Unknown type: {:?}", column.type_()); + SqlValue::Null + } + }; + + println!("{:?} {:?}", column.name(), value); + row_map.insert(column.name().to_string(), value); + } + result.rows.push(row_map); + } + + Ok(result) +} + +#[cfg_attr(mobile, tauri::mobile_entry_point)] +pub fn run() { + tauri::Builder::default() + .plugin(tauri_plugin_shell::init()) + .plugin(tauri_plugin_fs::init()) + .invoke_handler(tauri::generate_handler![pg_execute_query, greet]) + .run(tauri::generate_context!()) + .expect("error while running tauri application"); +} diff --git a/src-tauri/src/main.rs b/src-tauri/src/main.rs index 1703ba4..c672c6a 100644 --- a/src-tauri/src/main.rs +++ b/src-tauri/src/main.rs @@ -1,159 +1,5 @@ -// Prevents additional console window on Windows in release, DO NOT REMOVE!! #![cfg_attr(not(debug_assertions), windows_subsystem = "windows")] -use std::{ - any::Any, - collections::HashMap, - time::{Instant, SystemTime}, -}; - -use chrono::{DateTime, NaiveDateTime, Utc}; -use serde::{Deserialize, Serialize}; -use tauri::Manager; -use tokio_postgres::{ - types::{FromSql, Timestamp, ToSql, Type}, - Error, NoTls, -}; - -// Learn more about Tauri commands at https://tauri.app/v1/guides/features/command -#[tauri::command] -fn greet(name: &str) -> HashMap { - format!("Hello, {}! You've been greeted from Rust!", name); - let mut map: HashMap = HashMap::new(); - map.insert("name".to_string(), "John".to_string()); - map.insert("age".to_string(), "20".to_string()); - return map; -} - -#[derive(Debug, Serialize, Deserialize, Clone)] -#[serde(untagged)] -enum SqlValue { - String(Option), - Int(Option), - Float32(Option), - Float64(Option), - Bool(Option), - Timestamp(Option), - Null, -} - -#[derive(Debug, Serialize, Deserialize, Clone)] -struct ColInfo { - name: String, - col_type: String, - order_idx: i32, -} - -#[derive(Debug, Serialize, Deserialize, Clone)] -struct ExecuteQueryResult { - columns: Vec, - rows: Vec>, -} - -#[tauri::command] -async fn pg_execute_query( - host: String, - port: u16, - username: String, - password: String, - database: String, - query: String, - params: Vec, -) -> Result { - let connection_string = format!( - "postgres://{}:{}@{}:{}/{}", - username, password, host, port, database - ); - - // Connect to the database. - let (client, connection) = tokio_postgres::connect(&connection_string, NoTls) - .await - .map_err(|e| e.to_string())?; - - tokio::spawn(async move { - if let Err(e) = connection.await { - eprintln!("connection error: {}", e); - } - }); - - let params: Vec<&(dyn ToSql + Sync)> = params - .iter() - .map(|p| match p { - SqlValue::String(s) => s as &(dyn ToSql + Sync), - - SqlValue::Int(i) => i as &(dyn ToSql + Sync), - SqlValue::Float32(f) => f as &(dyn ToSql + Sync), - SqlValue::Float64(f) => f as &(dyn ToSql + Sync), - SqlValue::Bool(b) => b as &(dyn ToSql + Sync), - SqlValue::Timestamp(t) => t as &(dyn ToSql + Sync), - - SqlValue::Null => &"", - }) - .collect(); - - // Execute the query and fetch the results. - let rows = client - .query(&query, ¶ms) - .await - .map_err(|e| e.to_string())?; - - let mut result: ExecuteQueryResult = ExecuteQueryResult { - columns: Vec::new(), - rows: Vec::new(), - }; - - if rows.is_empty() { - return Ok(result); - } - - for (i, column) in rows[0].columns().iter().enumerate() { - result.columns.push(ColInfo { - name: column.name().to_string(), - col_type: column.type_().name().to_string(), - order_idx: i as i32, - }); - } - - for row in rows { - let mut row_map: HashMap = HashMap::new(); - for (i, column) in row.columns().iter().enumerate() { - let value = match column.type_() { - &Type::VARCHAR => SqlValue::String(row.get::<_, Option>(i)), - // &Type::TIMESTAMP => SqlValue::String(row.get::<_, Option>(i)), - &Type::TEXT => SqlValue::String(row.get::<_, Option>(i)), - &Type::NAME => SqlValue::String(row.get::<_, Option>(i)), - &Type::INT4 => SqlValue::Int(row.get::<_, Option>(i)), - &Type::BOOL => SqlValue::Bool(row.get::<_, Option>(i)), - &Type::FLOAT4 => SqlValue::Float32(row.get::<_, Option>(i)), - &Type::FLOAT8 => SqlValue::Float64(row.get::<_, Option>(i)), - &Type::TIMESTAMP => SqlValue::Timestamp(row.get::<_, Option>(i)), - - _ => { - println!(">>> Unknown type: {:?}", column.type_()); - SqlValue::Null - } - }; - - println!("{:?} {:?}", column.name(), value); - row_map.insert(column.name().to_string(), value); - } - result.rows.push(row_map); - } - - Ok(result) -} - fn main() { - tauri::Builder::default() - .setup(|app| { - #[cfg(debug_assertions)] // only include this code on debug builds - { - let window = app.get_window("main").unwrap(); - window.open_devtools(); - } - Ok(()) - }) - .invoke_handler(tauri::generate_handler![pg_execute_query, greet]) - .run(tauri::generate_context!()) - .expect("error while running tauri application"); + app_lib::run(); } diff --git a/src-tauri/tauri.conf.json b/src-tauri/tauri.conf.json index 26c3843..2307bc8 100644 --- a/src-tauri/tauri.conf.json +++ b/src-tauri/tauri.conf.json @@ -1,57 +1,37 @@ { - "$schema": "https://schema.tauri.app/config/1", - "build": { - "beforeDevCommand": "bun run dev", - "beforeBuildCommand": "bun run build", - "devPath": "http://localhost:1420", - "distDir": "../dist" + "$schema": "https://schema.tauri.app/config/2", + "build": { + "beforeDevCommand": "bun run dev", + "beforeBuildCommand": "bun run build", + "devUrl": "http://localhost:5173", + "frontendDist": "../dist" + }, + "bundle": { + "active": true, + "targets": "all", + "icon": [ + "icons/32x32.png", + "icons/128x128.png", + "icons/128x128@2x.png", + "icons/icon.icns", + "icons/icon.ico" + ] + }, + "productName": "database-browser", + "mainBinaryName": "database-browser", + "version": "0.0.3", + "identifier": "com.database-browser.app", + "plugins": {}, + "app": { + "security": { + "csp": null }, - "package": { - "productName": "database-browser", - "version": "0.0.3" - }, - "tauri": { - "allowlist": { - "all": false, - "shell": { - "all": false, - "open": true - }, - "window": { - "all": false, - "close": true, - "hide": true, - "show": true, - "maximize": true, - "minimize": true, - "unmaximize": true, - "unminimize": true, - "startDragging": true - } - }, - "windows": [ - { - "title": "database-browser", - "width": 1280, - "height": 800, - "decorations": false, - "transparent": true - } - ], - "security": { - "csp": null - }, - "bundle": { - "active": true, - "targets": "all", - "identifier": "com.database-browser.app", - "icon": [ - "icons/32x32.png", - "icons/128x128.png", - "icons/128x128@2x.png", - "icons/icon.icns", - "icons/icon.ico" - ] - } - } + "windows": [ + { + "title": "database-browser", + "width": 1280, + "height": 800 + } + ] + } } diff --git a/src/components/ButtonTabs.tsx b/src/components/ButtonTabs.tsx index 998767d..0373bab 100644 --- a/src/components/ButtonTabs.tsx +++ b/src/components/ButtonTabs.tsx @@ -1,4 +1,5 @@ import { Button, ButtonGroup } from "@blueprintjs/core"; +import { observer } from "mobx-react"; export interface ButtonTabEntry { name: string; @@ -13,24 +14,28 @@ export interface ButtonTabProps { isClosable?: boolean; } -export function ButtonTabs({ tabs, selectedTab, onSelectTab, onCloseTab, isClosable = true }: ButtonTabProps) { - return ( - - {tabs.map((tab) => ( - - ))} - - ); -} + {isClosable && ( + + ))} + + ); + }, +); diff --git a/src/components/ConnectionDialog.tsx b/src/components/ConnectionDialog.tsx index 6963faf..db857c0 100644 --- a/src/components/ConnectionDialog.tsx +++ b/src/components/ConnectionDialog.tsx @@ -1,16 +1,17 @@ +import type { ConnectionConfigFile } from "@/lib/connection-config-file"; import type { ConnectionArgs } from "@/lib/database"; import { assertExists } from "@/lib/utils"; +import { APP_MODEL } from "@/models/app"; import { Button, Dialog, DialogBody, DialogFooter, FormGroup, InputGroup, NumericInput, Tag } from "@blueprintjs/core"; import { useQuery } from "@tanstack/react-query"; import { observer } from "mobx-react"; import { useState } from "react"; import { useDebouncedValue } from "../hooks/use-debounced-value"; import { checkConnection } from "../lib/pgsql"; -import { CONNECTION_CONFIG_STORE, type ConnectionConfig } from "../models/connection-config"; interface ConnectionDialogProps { - connection?: ConnectionConfig; // If this is provided, the dialog will be in edit mode - onSubmit: (connection: ConnectionConfig) => void; + connection?: ConnectionConfigFile; // If this is provided, the dialog will be in edit mode + onSubmit: (connection: ConnectionConfigFile) => void; onClose: () => void; } @@ -18,12 +19,12 @@ export const ConnectionDialog = observer(({ connection, onSubmit, onClose }: Con const [connectionId] = useState(connection?.id || crypto.randomUUID()); // Form fields - const [connectionName, setConnectionName] = useState(connection?.name || ""); - const [databaseName, setDatabaseName] = useState(connection?.connection.database || ""); - const [databaseHost, setDatabaseHost] = useState(connection?.connection.host || ""); - const [databasePort, setDatabasePort] = useState(connection?.connection.port || ""); - const [databaseUsername, setDatabaseUsername] = useState(connection?.connection.user || ""); - const [databasePassword, setDatabasePassword] = useState(connection?.connection.password || ""); + const [connectionName, setConnectionName] = useState(connection?.name || "Local"); + const [databaseName, setDatabaseName] = useState(connection?.connection.database || "postgres"); + const [databaseHost, setDatabaseHost] = useState(connection?.connection.host || "localhost"); + const [databasePort, setDatabasePort] = useState(connection?.connection.port || 5432); + const [databaseUsername, setDatabaseUsername] = useState(connection?.connection.user || "postgres"); + const [databasePassword, setDatabasePassword] = useState(connection?.connection.password || "postgres"); // Resolve from default values. const resolvedHost = databaseHost || "localhost"; @@ -43,7 +44,7 @@ export const ConnectionDialog = observer(({ connection, onSubmit, onClose }: Con let nameError = connectionName.length === 0 && "Connection name is required"; nameError ||= - CONNECTION_CONFIG_STORE.configs.some((c) => c.name === connectionName && c.id !== connectionId) && + APP_MODEL.connections.some((c) => c.name === connectionName && c.id !== connectionId) && "Connection name must be unique"; const error = nameError; @@ -73,6 +74,10 @@ export const ConnectionDialog = observer(({ connection, onSubmit, onClose }: Con password: resolvedPassword, database: resolvedDatabase, }, + queries: [], + tags: [], + + order: 0, }); } @@ -102,7 +107,7 @@ export const ConnectionDialog = observer(({ connection, onSubmit, onClose }: Con - setDatabasePort(e.target.value)} /> + setDatabasePort(Number(e.target.value))} /> diff --git a/src/components/ManageConnectionsDialog.tsx b/src/components/ManageConnectionsDialog.tsx index 47c5b6e..eaaf275 100644 --- a/src/components/ManageConnectionsDialog.tsx +++ b/src/components/ManageConnectionsDialog.tsx @@ -1,24 +1,25 @@ import { Button, ButtonGroup, Dialog, DialogBody, HTMLTable, Icon, Tooltip } from "@blueprintjs/core"; import { checkConnection } from "@/lib/pgsql"; +import { APP_MODEL } from "@/models/app"; +import { ConnectionModel } from "@/models/connection"; import { useQuery } from "@tanstack/react-query"; import { observer } from "mobx-react"; import { useState } from "react"; -import { CONNECTION_CONFIG_STORE, ConnectionConfigModel } from "../models/connection-config"; import ConfirmIconButton from "./ConfirmIconButton"; import { ConnectionDialog } from "./ConnectionDialog"; import TrButton from "./TrButton"; interface ConnectionRowProps { - connection: ConnectionConfigModel; + connection: ConnectionModel; removeConnection: () => void; editConnection: () => void; - onSelectConnection: (connection: ConnectionConfigModel) => void; + onSelectConnection: (connection: ConnectionModel) => void; } const ConnectionRow = observer( ({ connection, removeConnection, editConnection, onSelectConnection }: ConnectionRowProps) => { - const url = `${connection.connection.host}:${connection.connection.port}`; + const url = `${connection.config.connection.host}:${connection.config.connection.port}`; const connectionStatusQuery = useQuery({ queryKey: ["connectionStatus", connection.id], @@ -26,7 +27,7 @@ const ConnectionRow = observer( retryDelay: 1000, refetchInterval: 10000, queryFn: async () => { - await checkConnection({ connection: connection.connection }); + await checkConnection({ connection: connection.config.connection }); return "ok"; }, }); @@ -34,7 +35,7 @@ const ConnectionRow = observer( onSelectConnection(connection)}> {connection.name} {url} - {connection.connection.user} + {connection.config.connection.user} {connectionStatusQuery.isLoading && } {connectionStatusQuery.isSuccess && } @@ -65,13 +66,13 @@ const ConnectionRow = observer( interface ManageConnectionsDialogProps { onClose: () => void; - onSelectConnection: (connection: ConnectionConfigModel) => void; + onSelectConnection: (connection: ConnectionModel) => void; } export const ManageConnectionsDialog = observer(({ onClose, onSelectConnection }: ManageConnectionsDialogProps) => { const [addConnectionDialogOpen, setAddConnectionDialogOpen] = useState(false); - const [editingConnection, setEditingConnection] = useState(null); + const [editingConnection, setEditingConnection] = useState(null); return ( @@ -90,12 +91,12 @@ export const ManageConnectionsDialog = observer(({ onClose, onSelectConnection } - {CONNECTION_CONFIG_STORE.configs.map((connection) => ( + {APP_MODEL.connections.map((connection) => ( { - CONNECTION_CONFIG_STORE.remove(connection.id); + APP_MODEL.removeConnection(connection.id); }} editConnection={() => { setEditingConnection(connection); @@ -108,7 +109,7 @@ export const ManageConnectionsDialog = observer(({ onClose, onSelectConnection } {editingConnection && ( { setEditingConnection(null); }} @@ -119,9 +120,7 @@ export const ManageConnectionsDialog = observer(({ onClose, onSelectConnection } {addConnectionDialogOpen && ( { - CONNECTION_CONFIG_STORE.add( - new ConnectionConfigModel(connection.id, connection.name, connection.connection), - ); + APP_MODEL.addConnection(new ConnectionModel(connection)); setAddConnectionDialogOpen(false); }} onClose={() => setAddConnectionDialogOpen(false)} diff --git a/src/components/Sidebar.tsx b/src/components/Sidebar.tsx new file mode 100644 index 0000000..c1d6d74 --- /dev/null +++ b/src/components/Sidebar.tsx @@ -0,0 +1,160 @@ +import type { ConnectionModel } from "@/models/connection"; +import { ContextMenu, Menu, MenuItem, Tag, Tree, type TreeNodeInfo } from "@blueprintjs/core"; +import { useMatchRoute, useNavigate } from "@tanstack/react-router"; +import { observer, useLocalObservable } from "mobx-react"; + +interface TreeData { + onClick?: () => void; +} + +interface SidebarProps { + connection: ConnectionModel; +} + +export const Sidebar = observer(({ connection }: SidebarProps) => { + const expanded = useLocalObservable(() => new Map()); + + const matchRoute = useMatchRoute(); + + const isDashboardSelected = matchRoute({ to: "/connection/$connectionId" }); + const isGraphSelected = matchRoute({ to: "/connection/$connectionId/visualizer" }); + const isCreateQuerySelected = matchRoute({ to: "/connection/$connectionId/query" }); + const isEditQuerySelected = matchRoute({ to: "/connection/$connectionId/query/$queryId" }); + const navigate = useNavigate(); + + const root: TreeNodeInfo[] = [ + { + id: "connection", + label: connection.name, + icon: "database", + isExpanded: expanded.get("connection") ?? true, + isSelected: !!isDashboardSelected, + nodeData: { + onClick: () => navigate({ to: "/connection/$connectionId", params: { connectionId: connection.id } }), + }, + + childNodes: [ + { + id: "visualize", + label: "Visualize", + icon: "graph", + isExpanded: expanded.get("visualize") ?? true, + isSelected: !!isGraphSelected, + nodeData: { + onClick: () => + navigate({ + to: "/connection/$connectionId/visualizer", + params: { connectionId: connection.id }, + }), + }, + }, + + { + id: "savedQueries", + label: ( + + Queries{" "} + + {connection.config.queries?.length ?? 0} + + + ), + icon: "antenna", + isExpanded: expanded.get("savedQueries") ?? true, + isSelected: !!isCreateQuerySelected, + nodeData: { + onClick: () => + navigate({ + to: "/connection/$connectionId/query", + params: { connectionId: connection.id }, + }), + }, + + childNodes: (connection.config.queries ?? []).map( + (query): TreeNodeInfo => ({ + id: query.id, + label: ( + + connection.removeQuery(query.id)} + text="Delete..." + intent="danger" + /> + + } + > + {query.name} + + ), + isSelected: !!matchRoute({ + to: "/connection/$connectionId/query/$queryId", + params: { + queryId: query.id, + }, + }), + nodeData: { + onClick: () => + navigate({ + to: "/connection/$connectionId/query/$queryId", + params: { connectionId: connection.id, queryId: query.id }, + }), + }, + }), + ), + }, + { + id: "tables", + label: ( + + Tables{" "} + + {connection.tables.length} + + + ), + icon: "panel-table", + isExpanded: expanded.get("tables") ?? true, + nodeData: { + onClick: () => + navigate({ + to: "/connection/$connectionId/table/", + params: { connectionId: connection.id }, + }), + }, + + childNodes: connection.tables.map( + (table): TreeNodeInfo => ({ + id: `${table.name}.${table.name}`, + label: table.name, + nodeData: { + onClick: () => + navigate({ + to: "/connection/$connectionId/table/$tableName", + params: { connectionId: connection.id, tableName: table.name }, + }), + }, + }), + ), + }, + ], + }, + ]; + + return ( + + onNodeClick={(node) => { + node.nodeData?.onClick?.(); + }} + onNodeCollapse={(node) => { + expanded.set(node.id as string, false); + }} + onNodeExpand={(node) => { + expanded.set(node.id as string, true); + }} + compact + contents={root} + /> + ); +}); diff --git a/src/components/SqlRowTable.tsx b/src/components/SqlRowTable.tsx index 44cad71..ffe509e 100644 --- a/src/components/SqlRowTable.tsx +++ b/src/components/SqlRowTable.tsx @@ -1,6 +1,5 @@ import { maybeGet, toMap, uniq } from "@/lib/utils"; import type { TableData } from "@/models/connection"; -import type {} from "@/models/connection-config"; import { H3, HTMLTable, Icon, Tag, Tooltip } from "@blueprintjs/core"; import { format } from "date-fns"; import { observer } from "mobx-react"; diff --git a/src/components/TagMultiSelect.tsx b/src/components/TagMultiSelect.tsx new file mode 100644 index 0000000..e4cb223 --- /dev/null +++ b/src/components/TagMultiSelect.tsx @@ -0,0 +1,42 @@ +import type { TagInfo } from "@/lib/connection-config-file"; +import { MenuItem, Tag } from "@blueprintjs/core"; +import { MultiSelect } from "@blueprintjs/select"; +import { useMemo, useState } from "react"; + +interface TagMultiSelectProps { + tags: TagInfo[]; + selectedTagIds: string[]; + onTagIdAdd: (tagId: string) => void; + onTagIdRemove: (tagId: string) => void; +} + +export const TagMultiSelect = (props: TagMultiSelectProps) => { + const { tags, selectedTagIds, onTagIdAdd, onTagIdRemove } = props; + + const selectedTags = useMemo(() => { + return tags.filter((tag) => selectedTagIds.includes(tag.id)); + }, [tags, selectedTagIds]); + + const [tagInput, setTagInput] = useState(""); + return ( + + itemRenderer={(tag, tagProps) => } + query={tagInput} + onQueryChange={(query) => setTagInput(query)} + tagRenderer={(tag) => {tag.name}} + itemPredicate={(query, item) => { + return item.name.toLowerCase().includes(query.toLowerCase()); + }} + items={tags} + selectedItems={selectedTags} + onItemSelect={(tag) => { + onTagIdAdd(tag.id); + setTagInput(""); + }} + onRemove={(tag) => { + onTagIdRemove(tag.id); + setTagInput(""); + }} + /> + ); +}; diff --git a/src/components/Titlebar.tsx b/src/components/Titlebar.tsx deleted file mode 100644 index 9c13a03..0000000 --- a/src/components/Titlebar.tsx +++ /dev/null @@ -1,38 +0,0 @@ -import { Button, ButtonGroup, Intent } from "@blueprintjs/core"; -import { appWindow } from "@tauri-apps/api/window"; -import { observer } from "mobx-react"; -import { SETTINGS_MODEL } from "./SettingsDialog"; - -export interface TitlebarProps { - title?: string; - children?: React.ReactNode; -} - -export const Titlebar = observer(({ title, children }: TitlebarProps) => { - return ( -
-
-
{title ?? document.title}
- {children} - - {SETTINGS_MODEL.values.isDebug && ( - {window.location.href} - )} -
-
-   -
- -
- ); -}); diff --git a/src/global.ts b/src/global.ts index d6d4734..37c9125 100644 --- a/src/global.ts +++ b/src/global.ts @@ -1,7 +1,13 @@ import { OverlayToaster } from "@blueprintjs/core"; import { QueryClient } from "@tanstack/react-query"; +import { createRoot } from "react-dom/client"; export const queryClient = new QueryClient(); -export const TOASTER = await OverlayToaster.createAsync({ position: "bottom" }); +export const TOASTER = await OverlayToaster.createAsync( + { position: "bottom" }, + { + domRenderer: (toaster, containerElement) => createRoot(containerElement).render(toaster), + }, +); export const IS_DEV_MODE = import.meta.env.DEV; diff --git a/src/layouts/RootLayout.tsx b/src/layouts/RootLayout.tsx new file mode 100644 index 0000000..5a88f52 --- /dev/null +++ b/src/layouts/RootLayout.tsx @@ -0,0 +1,79 @@ +import { type ButtonTabEntry, ButtonTabs } from "@/components/ButtonTabs"; +import { ManageConnectionsDialog } from "@/components/ManageConnectionsDialog"; +import { SETTINGS_MODEL, SettingsDialog } from "@/components/SettingsDialog"; +import { APP_MODEL } from "@/models/app"; +import { THEME_STORE, useTheme } from "@/models/theme"; +import { Button } from "@blueprintjs/core"; +import { ReactQueryDevtools } from "@tanstack/react-query-devtools"; +import { useLocation, useMatch, useNavigate } from "@tanstack/react-router"; +import { TanStackRouterDevtools } from "@tanstack/router-devtools"; +import { observer } from "mobx-react"; +import { useState } from "react"; + +interface RootLayoutProps { + children: React.ReactNode; +} + +export const RootLayout = observer(({ children }: RootLayoutProps) => { + useTheme(THEME_STORE); + + const [isManagingConnections, setIsManagingConnections] = useState(false); + const [isSettingsOpen, setIsSettingsOpen] = useState(false); + + const navigate = useNavigate(); + const match = useMatch({ from: "/connection/$connectionId/", shouldThrow: false }); + + const activeConnectionId = match?.params.connectionId; + + function setActiveConnectionId(id: string) { + navigate({ to: `/connection/${id}` }); + } + + const location = useLocation(); + const tabs: ButtonTabEntry[] = APP_MODEL.connections.map((conn) => ({ + name: conn.config.name, + value: conn.id, + })); + + return ( +
+
+ +
+ {location.pathname} + +
+
+ + {children} + + {isManagingConnections && ( + setIsManagingConnections(false)} + onSelectConnection={(connection) => { + // Find existing connection + const foundConn = APP_MODEL.connections.find((c) => c.id === connection.id); + + if (foundConn) { + setActiveConnectionId(foundConn.id); + } else { + setActiveConnectionId(connection.id); + } + + setIsManagingConnections(false); + }} + /> + )} + {isSettingsOpen && setIsSettingsOpen(false)} />} + + {SETTINGS_MODEL.values.isDebug && ( + <> + + + + )} +
+ ); +}); diff --git a/src/lib/connection-config-file.ts b/src/lib/connection-config-file.ts new file mode 100644 index 0000000..7f39593 --- /dev/null +++ b/src/lib/connection-config-file.ts @@ -0,0 +1,106 @@ +import { BaseDirectory, readFile, writeFile } from "@tauri-apps/plugin-fs"; +import { z } from "zod"; + +const ConnectionIdSchema = z.string(); +const TableIdSchema = z.string(); +const TagIdSchema = z.string(); + +export const QueryParamSchema = z.object({ + name: z.string(), + value: z.string(), + type: z.string(), +}); + +export const QuerySchema = z.object({ + id: z.string(), + + name: z.string(), + description: z.string(), + order: z.number(), + + tagIds: z + .array(TagIdSchema) + .optional() + .transform((tags) => tags ?? []), + + query: z.string(), + params: z + .array(QueryParamSchema) + .optional() + .transform((params) => params ?? []), +}); + +export type Query = z.infer; +export type NewQuery = Omit; + +export const ConnectionConfigSchema = z.object({ + host: z.string(), + port: z.number(), + user: z.string(), + password: z.string(), + database: z.string(), +}); + +export type ConnectionConfig = z.infer; + +export const TagInfoSchema = z.object({ + id: TagIdSchema, + name: z.string(), + description: z.string(), + order: z.number(), +}); + +export type TagInfo = z.infer; + +export const ConnectionConfigFileSchema = z.object({ + id: ConnectionIdSchema, + name: z.string(), + order: z.number(), + + connection: ConnectionConfigSchema, + + queries: z + .array(QuerySchema) + .optional() + .transform((queries) => queries ?? []), + + tags: z + .array(TagInfoSchema) + .optional() + .transform((tags) => tags ?? []), +}); + +export type ConnectionConfigFile = z.infer; + +export const AppConfigSchema = z.object({ + connections: z.array(ConnectionConfigFileSchema), +}); + +export type AppConfig = z.infer; + +export async function readConfigFile() { + try { + const configFileData = await readFile("config.json", { + baseDir: BaseDirectory.AppConfig, + }); + const decoder = new TextDecoder(); + const configFile = JSON.parse(decoder.decode(configFileData)); + const config = AppConfigSchema.parse(configFile); + return config; + } catch (error) { + console.error(error); + return null; + } +} + +export async function writeConfigFile(config: AppConfig) { + try { + const encoder = new TextEncoder(); + const data = encoder.encode(JSON.stringify(config, null, 2)); + await writeFile("config.json", data, { + baseDir: BaseDirectory.AppConfig, + }); + } catch (error) { + console.error(error); + } +} diff --git a/src/lib/pgsql.ts b/src/lib/pgsql.ts index c8b16f9..0a3cae4 100644 --- a/src/lib/pgsql.ts +++ b/src/lib/pgsql.ts @@ -1,4 +1,4 @@ -import { invoke } from "@tauri-apps/api/tauri"; +import { invoke } from "@tauri-apps/api/core"; import type { ConnectionArgs } from "@/lib/database"; import { z } from "zod"; diff --git a/src/lib/utils.ts b/src/lib/utils.ts index 3ddcae5..788c1ba 100644 --- a/src/lib/utils.ts +++ b/src/lib/utils.ts @@ -66,14 +66,18 @@ export function clamp(value: number, min: number, max: number) { return Math.min(Math.max(value, min), max); } -export interface BaseDiscUnion { - type: string; +export interface BaseDiscUnion { + type: T; } -export function matchUnions< - T extends BaseDiscUnion, - U extends { [K in T["type"]]: (value: Extract) => unknown }, ->(value: T, cases: U): ReturnType { +type DiscUnionMatchObj> = { + [K in T["type"]]: (value: Extract) => unknown; +}; + +export function matchUnions, U extends DiscUnionMatchObj>( + value: T, + cases: U, +): ReturnType { assertKeyOfObject(cases, value.type); return cases[value.type as keyof U](value as any) as any; } diff --git a/src/main.tsx b/src/main.tsx index fe577b1..012b281 100644 --- a/src/main.tsx +++ b/src/main.tsx @@ -26,11 +26,11 @@ declare module "@tanstack/react-router" { // console.log(test); import { assertExists } from "@/lib/utils"; -import { BlueprintProvider, Intent } from "@blueprintjs/core"; +import { BlueprintProvider } from "@blueprintjs/core"; import { QueryClientProvider } from "@tanstack/react-query"; import { RouterProvider, createRouter } from "@tanstack/react-router"; import { observer } from "mobx-react"; -import { TOASTER, queryClient } from "./global"; +import { queryClient } from "./global"; const rootEl = document.getElementById("root"); assertExists(rootEl); @@ -49,6 +49,6 @@ const Root = observer(() => { const root = createRoot(rootEl); root.render(); -window.addEventListener("error", (event) => { - TOASTER.show({ intent: Intent.DANGER, message: event.error.message }); -}); +// window.addEventListener("error", (event) => { +// TOASTER.show({ intent: Intent.DANGER, message: event.error.message }); +// }); diff --git a/src/models/app.tsx b/src/models/app.tsx new file mode 100644 index 0000000..3590c61 --- /dev/null +++ b/src/models/app.tsx @@ -0,0 +1,62 @@ +import { type AppConfig, AppConfigSchema, readConfigFile, writeConfigFile } from "@/lib/connection-config-file"; +import { autorun, makeAutoObservable } from "mobx"; +import { ConnectionModel } from "./connection"; + +export class AppModel { + connections: ConnectionModel[] = []; + + constructor() { + makeAutoObservable(this); + } + + async readConfig() { + const config = await readConfigFile(); + if (!config) return null; + const result = AppConfigSchema.safeParse(config); + if (!result.success) return null; + return result.data; + } + + async loadConfig() { + console.log("Loading config"); + const config = await this.readConfig(); + if (!config) return; + this.connections = config.connections.map((c) => new ConnectionModel(c)); + } + + listenAndSave() { + autorun( + () => { + console.log("saving config"); + this.saveConfig(); + }, + { + delay: 1000, + }, + ); + } + + async saveConfig() { + console.log("saving config"); + const config: AppConfig = { + connections: this.connections.map((c) => c.config), + }; + await writeConfigFile(config); + } + + getConnection(id: string) { + return this.connections.find((c) => c.id === id); + } + + async addConnection(connection: ConnectionModel) { + this.connections.push(connection); + } + + async removeConnection(id: string) { + this.connections = this.connections.filter((c) => c.id !== id); + } +} + +export const APP_MODEL = new AppModel(); +await APP_MODEL.loadConfig().catch(console.error); +APP_MODEL.listenAndSave(); diff --git a/src/models/connection-config.tsx b/src/models/connection-config.tsx deleted file mode 100644 index 1b8003e..0000000 --- a/src/models/connection-config.tsx +++ /dev/null @@ -1,83 +0,0 @@ -import { type ConnectionArgs, ConnectionArgsSchema } from "@/lib/database"; -import { zodLocalStorageGetItem, zodLocalStorageSetItem } from "@/lib/zod"; -import { autorun, makeAutoObservable } from "mobx"; -import { z } from "zod"; - -const ConnectionConfigSchema = z.object({ - id: z.string(), - name: z.string(), - connection: ConnectionArgsSchema, -}); - -export type ConnectionConfig = z.infer; - -export const ConnectionConfigStoreSchema = z.object({ - configs: z.array(ConnectionConfigSchema), -}); -export type ConnectionConfigStore = z.infer; - -export class ConnectionConfigModel { - id: string; - name: string; - connection: ConnectionArgs; - - constructor(id: string, name: string, connection: ConnectionArgs) { - this.id = id; - this.name = name; - this.connection = connection; - } -} - -export enum DatabaseConnectionStatus { - IDLE = "IDLE", - CONNECTED = "CONNECTED", - CONNECTING = "CONNECTING", - FAILED = "FAILED", -} - -export class ConnectionConfigModelStore { - configs: ConnectionConfigModel[] = []; - - constructor() { - const data = zodLocalStorageGetItem("connection-configs", ConnectionConfigStoreSchema); - this.configs = data?.configs ?? [ - new ConnectionConfigModel("default", "Default", { - host: "localhost", - port: 5432, - user: "postgres", - password: "postgres", - database: "postgres", - }), - new ConnectionConfigModel("test", "Test", { - host: "localhost", - port: 5432, - user: "postgres", - password: "postgres", - database: "postgres", - }), - ]; - - makeAutoObservable(this); - - autorun( - () => { - zodLocalStorageSetItem("connection-configs", ConnectionConfigStoreSchema, { - configs: this.configs, - }); - }, - { - delay: 1000, - }, - ); - } - - add(config: ConnectionConfigModel) { - this.configs.push(config); - } - - remove(id: string) { - this.configs = this.configs.filter((config) => config.id !== id); - } -} - -export const CONNECTION_CONFIG_STORE = new ConnectionConfigModelStore(); diff --git a/src/models/connection.tsx b/src/models/connection.tsx index cb82de6..5520e30 100644 --- a/src/models/connection.tsx +++ b/src/models/connection.tsx @@ -1,4 +1,5 @@ import { promiseAll } from "@/lib/async"; +import type { ConnectionConfigFile, NewQuery, Query } from "@/lib/connection-config-file"; import { makeAutoObservable, runInAction } from "mobx"; import { type ColumnInfo, @@ -15,7 +16,6 @@ import { getTables, getUniqueKeys, } from "../lib/pgsql"; -import type { ConnectionConfigModel } from "./connection-config"; export enum DatabaseConnectionStatus { IDLE = "IDLE", @@ -38,7 +38,7 @@ export interface TableData { } export class ConnectionModel { - config: ConnectionConfigModel; + config: ConnectionConfigFile; tables: TableInfo[] = []; columns: ColumnInfo[] = []; @@ -48,7 +48,7 @@ export class ConnectionModel { status = DatabaseConnectionStatus.IDLE; - constructor(config: ConnectionConfigModel) { + constructor(config: ConnectionConfigFile) { this.config = config; makeAutoObservable(this); @@ -62,6 +62,14 @@ export class ConnectionModel { return this.config.name; } + getTag(id: string) { + return this.config.tags.find((t) => t.id === id); + } + + getQuery(id: string) { + return this.config.queries.find((q) => q.id === id); + } + async connect() { this.status = DatabaseConnectionStatus.CONNECTING; try { @@ -74,6 +82,27 @@ export class ConnectionModel { } } + updateQuery(query: NewQuery | Query) { + if ("id" in query && query.id != null) { + // Found existing query, update it. + this.config.queries = this.config.queries.map((q) => (q.id === query.id ? query : q)); + return query; + } + + // New query, create it. + const newQuery: Query = { + ...query, + id: crypto.randomUUID(), + }; + this.config.queries.push(newQuery); + + return newQuery; + } + + removeQuery(id: string) { + this.config.queries = this.config.queries.filter((q) => q.id !== id); + } + async updateMeta() { try { const result = await promiseAll({ @@ -175,30 +204,3 @@ export class ConnectionModel { return { data: data.rows, sql, colInfo, query: data }; } } -export class DatabaseConnectionModelStore { - connections: ConnectionModel[] = []; - - constructor() { - makeAutoObservable(this); - } - - removeConnection(id: string) { - this.connections = this.connections.filter((connection) => connection.id !== id); - } -} - -export const databaseConnectionModelStore = new DatabaseConnectionModelStore(); - -databaseConnectionModelStore.connections.push( - new ConnectionModel({ - id: "1", - name: "Local", - connection: { - host: "localhost", - port: 5432, - database: "postgres", - user: "postgres", - password: "postgres", - }, - }), -); diff --git a/src/models/model-pool.tsx b/src/models/model-pool.tsx new file mode 100644 index 0000000..29bb6f0 --- /dev/null +++ b/src/models/model-pool.tsx @@ -0,0 +1,33 @@ +import { makeAutoObservable } from "mobx"; + +interface ModelPoolEntry { + id: string; + model: T; +} + +interface ModelPoolOptions { + fetch: (id: string) => Promise; +} + +export class ModelPool { + private readonly fetchFn: (id: string) => Promise; + + private pool: Map = new Map(); + + constructor(options: ModelPoolOptions) { + this.fetchFn = options.fetch; + + makeAutoObservable(this); + } + + async get(id: string) { + const model = this.pool.get(id); + if (model) { + return model; + } + + const newModel = await this.fetchFn(id); + this.pool.set(id, newModel); + return newModel; + } +} diff --git a/src/routeTree.gen.ts b/src/routeTree.gen.ts index efaf1a6..ffdd0c3 100644 --- a/src/routeTree.gen.ts +++ b/src/routeTree.gen.ts @@ -13,10 +13,29 @@ import { createFileRoute } from '@tanstack/react-router' // Import Routes import { Route as rootRoute } from './routes/__root' +import { Route as ConnectionConnectionIdImport } from './routes/connection/$connectionId' // Create Virtual Routes const IndexLazyImport = createFileRoute('/')() +const ConnectionConnectionIdIndexLazyImport = createFileRoute( + '/connection/$connectionId/', +)() +const ConnectionConnectionIdVisualizerLazyImport = createFileRoute( + '/connection/$connectionId/visualizer', +)() +const ConnectionConnectionIdTableIndexLazyImport = createFileRoute( + '/connection/$connectionId/table/', +)() +const ConnectionConnectionIdQueryIndexLazyImport = createFileRoute( + '/connection/$connectionId/query/', +)() +const ConnectionConnectionIdTableTableNameIndexLazyImport = createFileRoute( + '/connection/$connectionId/table/$tableName/', +)() +const ConnectionConnectionIdQueryQueryIdIndexLazyImport = createFileRoute( + '/connection/$connectionId/query/$queryId/', +)() // Create/Update Routes @@ -25,6 +44,69 @@ const IndexLazyRoute = IndexLazyImport.update({ getParentRoute: () => rootRoute, } as any).lazy(() => import('./routes/index.lazy').then((d) => d.Route)) +const ConnectionConnectionIdRoute = ConnectionConnectionIdImport.update({ + path: '/connection/$connectionId', + getParentRoute: () => rootRoute, +} as any) + +const ConnectionConnectionIdIndexLazyRoute = + ConnectionConnectionIdIndexLazyImport.update({ + path: '/', + getParentRoute: () => ConnectionConnectionIdRoute, + } as any).lazy(() => + import('./routes/connection/$connectionId/index.lazy').then((d) => d.Route), + ) + +const ConnectionConnectionIdVisualizerLazyRoute = + ConnectionConnectionIdVisualizerLazyImport.update({ + path: '/visualizer', + getParentRoute: () => ConnectionConnectionIdRoute, + } as any).lazy(() => + import('./routes/connection/$connectionId/visualizer.lazy').then( + (d) => d.Route, + ), + ) + +const ConnectionConnectionIdTableIndexLazyRoute = + ConnectionConnectionIdTableIndexLazyImport.update({ + path: '/table/', + getParentRoute: () => ConnectionConnectionIdRoute, + } as any).lazy(() => + import('./routes/connection/$connectionId/table/index.lazy').then( + (d) => d.Route, + ), + ) + +const ConnectionConnectionIdQueryIndexLazyRoute = + ConnectionConnectionIdQueryIndexLazyImport.update({ + path: '/query/', + getParentRoute: () => ConnectionConnectionIdRoute, + } as any).lazy(() => + import('./routes/connection/$connectionId/query/index.lazy').then( + (d) => d.Route, + ), + ) + +const ConnectionConnectionIdTableTableNameIndexLazyRoute = + ConnectionConnectionIdTableTableNameIndexLazyImport.update({ + path: '/table/$tableName/', + getParentRoute: () => ConnectionConnectionIdRoute, + } as any).lazy(() => + import( + './routes/connection/$connectionId/table/$tableName/index.lazy' + ).then((d) => d.Route), + ) + +const ConnectionConnectionIdQueryQueryIdIndexLazyRoute = + ConnectionConnectionIdQueryQueryIdIndexLazyImport.update({ + path: '/query/$queryId/', + getParentRoute: () => ConnectionConnectionIdRoute, + } as any).lazy(() => + import('./routes/connection/$connectionId/query/$queryId/index.lazy').then( + (d) => d.Route, + ), + ) + // Populate the FileRoutesByPath interface declare module '@tanstack/react-router' { @@ -36,39 +118,163 @@ declare module '@tanstack/react-router' { preLoaderRoute: typeof IndexLazyImport parentRoute: typeof rootRoute } + '/connection/$connectionId': { + id: '/connection/$connectionId' + path: '/connection/$connectionId' + fullPath: '/connection/$connectionId' + preLoaderRoute: typeof ConnectionConnectionIdImport + parentRoute: typeof rootRoute + } + '/connection/$connectionId/visualizer': { + id: '/connection/$connectionId/visualizer' + path: '/visualizer' + fullPath: '/connection/$connectionId/visualizer' + preLoaderRoute: typeof ConnectionConnectionIdVisualizerLazyImport + parentRoute: typeof ConnectionConnectionIdImport + } + '/connection/$connectionId/': { + id: '/connection/$connectionId/' + path: '/' + fullPath: '/connection/$connectionId/' + preLoaderRoute: typeof ConnectionConnectionIdIndexLazyImport + parentRoute: typeof ConnectionConnectionIdImport + } + '/connection/$connectionId/query/': { + id: '/connection/$connectionId/query/' + path: '/query' + fullPath: '/connection/$connectionId/query' + preLoaderRoute: typeof ConnectionConnectionIdQueryIndexLazyImport + parentRoute: typeof ConnectionConnectionIdImport + } + '/connection/$connectionId/table/': { + id: '/connection/$connectionId/table/' + path: '/table' + fullPath: '/connection/$connectionId/table' + preLoaderRoute: typeof ConnectionConnectionIdTableIndexLazyImport + parentRoute: typeof ConnectionConnectionIdImport + } + '/connection/$connectionId/query/$queryId/': { + id: '/connection/$connectionId/query/$queryId/' + path: '/query/$queryId' + fullPath: '/connection/$connectionId/query/$queryId' + preLoaderRoute: typeof ConnectionConnectionIdQueryQueryIdIndexLazyImport + parentRoute: typeof ConnectionConnectionIdImport + } + '/connection/$connectionId/table/$tableName/': { + id: '/connection/$connectionId/table/$tableName/' + path: '/table/$tableName' + fullPath: '/connection/$connectionId/table/$tableName' + preLoaderRoute: typeof ConnectionConnectionIdTableTableNameIndexLazyImport + parentRoute: typeof ConnectionConnectionIdImport + } } } // Create and export the route tree +interface ConnectionConnectionIdRouteChildren { + ConnectionConnectionIdVisualizerLazyRoute: typeof ConnectionConnectionIdVisualizerLazyRoute + ConnectionConnectionIdIndexLazyRoute: typeof ConnectionConnectionIdIndexLazyRoute + ConnectionConnectionIdQueryIndexLazyRoute: typeof ConnectionConnectionIdQueryIndexLazyRoute + ConnectionConnectionIdTableIndexLazyRoute: typeof ConnectionConnectionIdTableIndexLazyRoute + ConnectionConnectionIdQueryQueryIdIndexLazyRoute: typeof ConnectionConnectionIdQueryQueryIdIndexLazyRoute + ConnectionConnectionIdTableTableNameIndexLazyRoute: typeof ConnectionConnectionIdTableTableNameIndexLazyRoute +} + +const ConnectionConnectionIdRouteChildren: ConnectionConnectionIdRouteChildren = + { + ConnectionConnectionIdVisualizerLazyRoute: + ConnectionConnectionIdVisualizerLazyRoute, + ConnectionConnectionIdIndexLazyRoute: ConnectionConnectionIdIndexLazyRoute, + ConnectionConnectionIdQueryIndexLazyRoute: + ConnectionConnectionIdQueryIndexLazyRoute, + ConnectionConnectionIdTableIndexLazyRoute: + ConnectionConnectionIdTableIndexLazyRoute, + ConnectionConnectionIdQueryQueryIdIndexLazyRoute: + ConnectionConnectionIdQueryQueryIdIndexLazyRoute, + ConnectionConnectionIdTableTableNameIndexLazyRoute: + ConnectionConnectionIdTableTableNameIndexLazyRoute, + } + +const ConnectionConnectionIdRouteWithChildren = + ConnectionConnectionIdRoute._addFileChildren( + ConnectionConnectionIdRouteChildren, + ) + export interface FileRoutesByFullPath { '/': typeof IndexLazyRoute + '/connection/$connectionId': typeof ConnectionConnectionIdRouteWithChildren + '/connection/$connectionId/visualizer': typeof ConnectionConnectionIdVisualizerLazyRoute + '/connection/$connectionId/': typeof ConnectionConnectionIdIndexLazyRoute + '/connection/$connectionId/query': typeof ConnectionConnectionIdQueryIndexLazyRoute + '/connection/$connectionId/table': typeof ConnectionConnectionIdTableIndexLazyRoute + '/connection/$connectionId/query/$queryId': typeof ConnectionConnectionIdQueryQueryIdIndexLazyRoute + '/connection/$connectionId/table/$tableName': typeof ConnectionConnectionIdTableTableNameIndexLazyRoute } export interface FileRoutesByTo { '/': typeof IndexLazyRoute + '/connection/$connectionId/visualizer': typeof ConnectionConnectionIdVisualizerLazyRoute + '/connection/$connectionId': typeof ConnectionConnectionIdIndexLazyRoute + '/connection/$connectionId/query': typeof ConnectionConnectionIdQueryIndexLazyRoute + '/connection/$connectionId/table': typeof ConnectionConnectionIdTableIndexLazyRoute + '/connection/$connectionId/query/$queryId': typeof ConnectionConnectionIdQueryQueryIdIndexLazyRoute + '/connection/$connectionId/table/$tableName': typeof ConnectionConnectionIdTableTableNameIndexLazyRoute } export interface FileRoutesById { __root__: typeof rootRoute '/': typeof IndexLazyRoute + '/connection/$connectionId': typeof ConnectionConnectionIdRouteWithChildren + '/connection/$connectionId/visualizer': typeof ConnectionConnectionIdVisualizerLazyRoute + '/connection/$connectionId/': typeof ConnectionConnectionIdIndexLazyRoute + '/connection/$connectionId/query/': typeof ConnectionConnectionIdQueryIndexLazyRoute + '/connection/$connectionId/table/': typeof ConnectionConnectionIdTableIndexLazyRoute + '/connection/$connectionId/query/$queryId/': typeof ConnectionConnectionIdQueryQueryIdIndexLazyRoute + '/connection/$connectionId/table/$tableName/': typeof ConnectionConnectionIdTableTableNameIndexLazyRoute } export interface FileRouteTypes { fileRoutesByFullPath: FileRoutesByFullPath - fullPaths: '/' + fullPaths: + | '/' + | '/connection/$connectionId' + | '/connection/$connectionId/visualizer' + | '/connection/$connectionId/' + | '/connection/$connectionId/query' + | '/connection/$connectionId/table' + | '/connection/$connectionId/query/$queryId' + | '/connection/$connectionId/table/$tableName' fileRoutesByTo: FileRoutesByTo - to: '/' - id: '__root__' | '/' + to: + | '/' + | '/connection/$connectionId/visualizer' + | '/connection/$connectionId' + | '/connection/$connectionId/query' + | '/connection/$connectionId/table' + | '/connection/$connectionId/query/$queryId' + | '/connection/$connectionId/table/$tableName' + id: + | '__root__' + | '/' + | '/connection/$connectionId' + | '/connection/$connectionId/visualizer' + | '/connection/$connectionId/' + | '/connection/$connectionId/query/' + | '/connection/$connectionId/table/' + | '/connection/$connectionId/query/$queryId/' + | '/connection/$connectionId/table/$tableName/' fileRoutesById: FileRoutesById } export interface RootRouteChildren { IndexLazyRoute: typeof IndexLazyRoute + ConnectionConnectionIdRoute: typeof ConnectionConnectionIdRouteWithChildren } const rootRouteChildren: RootRouteChildren = { IndexLazyRoute: IndexLazyRoute, + ConnectionConnectionIdRoute: ConnectionConnectionIdRouteWithChildren, } export const routeTree = rootRoute @@ -83,11 +289,47 @@ export const routeTree = rootRoute "__root__": { "filePath": "__root.tsx", "children": [ - "/" + "/", + "/connection/$connectionId" ] }, "/": { "filePath": "index.lazy.tsx" + }, + "/connection/$connectionId": { + "filePath": "connection/$connectionId.tsx", + "children": [ + "/connection/$connectionId/visualizer", + "/connection/$connectionId/", + "/connection/$connectionId/query/", + "/connection/$connectionId/table/", + "/connection/$connectionId/query/$queryId/", + "/connection/$connectionId/table/$tableName/" + ] + }, + "/connection/$connectionId/visualizer": { + "filePath": "connection/$connectionId/visualizer.lazy.tsx", + "parent": "/connection/$connectionId" + }, + "/connection/$connectionId/": { + "filePath": "connection/$connectionId/index.lazy.tsx", + "parent": "/connection/$connectionId" + }, + "/connection/$connectionId/query/": { + "filePath": "connection/$connectionId/query/index.lazy.tsx", + "parent": "/connection/$connectionId" + }, + "/connection/$connectionId/table/": { + "filePath": "connection/$connectionId/table/index.lazy.tsx", + "parent": "/connection/$connectionId" + }, + "/connection/$connectionId/query/$queryId/": { + "filePath": "connection/$connectionId/query/$queryId/index.lazy.tsx", + "parent": "/connection/$connectionId" + }, + "/connection/$connectionId/table/$tableName/": { + "filePath": "connection/$connectionId/table/$tableName/index.lazy.tsx", + "parent": "/connection/$connectionId" } } } diff --git a/src/routes/__root.tsx b/src/routes/__root.tsx index 2bf2d1c..3f81a41 100644 --- a/src/routes/__root.tsx +++ b/src/routes/__root.tsx @@ -1,28 +1,14 @@ -import { SETTINGS_MODEL } from "@/components/SettingsDialog"; -import { Titlebar } from "@/components/Titlebar"; -import { THEME_STORE, useTheme } from "@/models/theme"; -import { ReactQueryDevtools } from "@tanstack/react-query-devtools"; +import { RootLayout } from "@/layouts/RootLayout"; import { Outlet, createRootRoute } from "@tanstack/react-router"; -import { TanStackRouterDevtools } from "@tanstack/router-devtools"; import { observer } from "mobx-react"; export const Route = createRootRoute({ + notFoundComponent: (props) =>
Not Found
, component: observer(() => { - useTheme(THEME_STORE); - return ( -
- - + - - {SETTINGS_MODEL.values.isDebug && ( - <> - - - - )} -
+ ); }), }); diff --git a/src/routes/connection/$connectionId.tsx b/src/routes/connection/$connectionId.tsx new file mode 100644 index 0000000..c038e12 --- /dev/null +++ b/src/routes/connection/$connectionId.tsx @@ -0,0 +1,39 @@ +import { Sidebar } from "@/components/Sidebar"; +import { SidebarLayout } from "@/components/SidebarLayout"; +import { APP_MODEL } from "@/models/app"; +import { useQuery } from "@tanstack/react-query"; +import { Outlet, createFileRoute, notFound } from "@tanstack/react-router"; +import { observer } from "mobx-react"; +import { ErrorBoundary } from "react-error-boundary"; + +export const Route = createFileRoute("/connection/$connectionId")({ + loader: (ctx) => { + const connection = APP_MODEL.getConnection(ctx.params.connectionId); + if (!connection) { + throw notFound(); + } + + return { + connection: connection, + }; + }, + component: observer(Component), +}); + +function Component() { + const { connection } = Route.useLoaderData(); + + const metaQuery = useQuery({ + queryKey: ["meta", connection.id], + queryFn: () => connection.updateMeta(), + refetchInterval: 1000 * 10, + }); + + return ( + }> +
Error: {JSON.stringify(props.error)}
}> + +
+
+ ); +} diff --git a/src/routes/connection/$connectionId/index.lazy.tsx b/src/routes/connection/$connectionId/index.lazy.tsx new file mode 100644 index 0000000..0b387b2 --- /dev/null +++ b/src/routes/connection/$connectionId/index.lazy.tsx @@ -0,0 +1,16 @@ +import { APP_MODEL } from "@/models/app"; +import { ConnectionInfoView } from "@/views/ConnectionInfoView"; +import { createLazyFileRoute } from "@tanstack/react-router"; +import { observer } from "mobx-react"; + +export const Route = createLazyFileRoute("/connection/$connectionId/")({ + component: observer(Component), +}); + +function Component() { + const { connectionId } = Route.useParams(); + const connection = APP_MODEL.getConnection(connectionId); + if (!connection) return
Connection Not Found: {connectionId}
; + + return ; +} diff --git a/src/routes/connection/$connectionId/query/$queryId/index.lazy.tsx b/src/routes/connection/$connectionId/query/$queryId/index.lazy.tsx new file mode 100644 index 0000000..5a3ca27 --- /dev/null +++ b/src/routes/connection/$connectionId/query/$queryId/index.lazy.tsx @@ -0,0 +1,27 @@ +import { APP_MODEL } from "@/models/app"; +import { SavedQueryEditorView } from "@/views/SavedQueryView"; +import { createLazyFileRoute } from "@tanstack/react-router"; +import { observer } from "mobx-react"; + +export const Route = createLazyFileRoute("/connection/$connectionId/query/$queryId/")({ + component: observer(Component), +}); + +function Component() { + const { connectionId, queryId } = Route.useParams(); + const connection = APP_MODEL.getConnection(connectionId); + if (!connection) return
Connection Not Found: {connectionId}
; + + const query = connection.getQuery(queryId); + if (!query) return
Query Not Found: {queryId}
; + + return ( + { + return connection.updateQuery(q); + }} + /> + ); +} diff --git a/src/routes/connection/$connectionId/query/index.lazy.tsx b/src/routes/connection/$connectionId/query/index.lazy.tsx new file mode 100644 index 0000000..eb9d024 --- /dev/null +++ b/src/routes/connection/$connectionId/query/index.lazy.tsx @@ -0,0 +1,24 @@ +import { APP_MODEL } from "@/models/app"; +import { SavedQueryEditorView } from "@/views/SavedQueryView"; +import { createLazyFileRoute } from "@tanstack/react-router"; +import { observer } from "mobx-react"; + +export const Route = createLazyFileRoute("/connection/$connectionId/query/")({ + component: observer(Component), +}); + +function Component() { + const { connectionId } = Route.useParams(); + + const connection = APP_MODEL.getConnection(connectionId); + if (!connection) return
Connection Not Found: {connectionId}
; + + return ( + { + return connection.updateQuery(q); + }} + /> + ); +} diff --git a/src/routes/connection/$connectionId/table/$tableName/index.lazy.tsx b/src/routes/connection/$connectionId/table/$tableName/index.lazy.tsx new file mode 100644 index 0000000..1e22f29 --- /dev/null +++ b/src/routes/connection/$connectionId/table/$tableName/index.lazy.tsx @@ -0,0 +1,19 @@ +import { APP_MODEL } from "@/models/app"; +import { TableBrowserView } from "@/views/TableBrowserView"; +import { createLazyFileRoute } from "@tanstack/react-router"; +import { observer } from "mobx-react"; + +export const Route = createLazyFileRoute("/connection/$connectionId/table/$tableName/")({ + component: observer(Component), +}); + +function Component() { + const { connectionId, tableName } = Route.useParams(); + const connection = APP_MODEL.getConnection(connectionId); + if (!connection) return
Connection Not Found: {connectionId}
; + + const entry = connection.tables.find((table) => table.name === tableName); + if (!entry) return
Table Not Found: {tableName}
; + + return ; +} diff --git a/src/routes/connection/$connectionId/table/index.lazy.tsx b/src/routes/connection/$connectionId/table/index.lazy.tsx new file mode 100644 index 0000000..0e45c98 --- /dev/null +++ b/src/routes/connection/$connectionId/table/index.lazy.tsx @@ -0,0 +1,5 @@ +import { createLazyFileRoute } from "@tanstack/react-router"; + +export const Route = createLazyFileRoute("/connection/$connectionId/table/")({ + component: () =>
Hello /connection/$connectionId/table/!
, +}); diff --git a/src/routes/connection/$connectionId/visualizer.lazy.tsx b/src/routes/connection/$connectionId/visualizer.lazy.tsx new file mode 100644 index 0000000..bbeec88 --- /dev/null +++ b/src/routes/connection/$connectionId/visualizer.lazy.tsx @@ -0,0 +1,18 @@ +import { APP_MODEL } from "@/models/app"; +import { TableNodeView } from "@/views/TableNodeView"; +import { createLazyFileRoute, notFound } from "@tanstack/react-router"; +import { observer } from "mobx-react"; + +export const Route = createLazyFileRoute("/connection/$connectionId/visualizer")({ + component: observer(Component), + notFoundComponent: () =>
Not Found
, +}); + +function Component() { + const { connectionId } = Route.useParams(); + const connection = APP_MODEL.getConnection(connectionId); + if (!connection) { + throw notFound(); + } + return ; +} diff --git a/src/routes/index.lazy.tsx b/src/routes/index.lazy.tsx index 9e3ac2b..5dd8516 100644 --- a/src/routes/index.lazy.tsx +++ b/src/routes/index.lazy.tsx @@ -1,69 +1,11 @@ -import { ButtonTabs } from "@/components/ButtonTabs"; -import { ManageConnectionsDialog } from "@/components/ManageConnectionsDialog"; -import { SettingsDialog } from "@/components/SettingsDialog"; -import { ConnectionModel } from "@/models/connection"; -import { ConnectionBrowser } from "@/views/ConnectionView"; -import { Button } from "@blueprintjs/core"; import { createLazyFileRoute } from "@tanstack/react-router"; import { observer } from "mobx-react"; -import { useState } from "react"; export const Route = createLazyFileRoute("/")({ component: observer(() => { - const [isManagingConnections, setIsManagingConnections] = useState(false); - const [isSettingsOpen, setIsSettingsOpen] = useState(false); - - const [activeConnections, setActiveConnections] = useState([]); - const [activeConnectionId, setActiveConnectionId] = useState(null); - - const tabs = activeConnections.map((connection) => ({ - name: connection.name, - value: connection.id, - })); - - const activeConnection = activeConnections.find((c) => c.id === activeConnectionId); - return ( <> -
- { - const connection = activeConnections.find((c) => c.id === newTabId); - if (!connection) return; - - setActiveConnectionId(connection.id); - }} - onCloseTab={(tabId) => { - setActiveConnections(activeConnections.filter((c) => c.id !== tabId)); - }} - /> -
-
-
- {activeConnection && } - {isManagingConnections && ( - setIsManagingConnections(false)} - onSelectConnection={(connection) => { - // Find existing connection - const foundConn = activeConnections.find((c) => c.config.id === connection.id); - - if (foundConn) { - setActiveConnectionId(foundConn.id); - } else { - setActiveConnections([...activeConnections, new ConnectionModel(connection)]); - setActiveConnectionId(connection.id); - } - - setIsManagingConnections(false); - }} - /> - )} - {isSettingsOpen && setIsSettingsOpen(false)} />} +
Select a connection!
); }), diff --git a/src/styles/index.css b/src/styles/index.css index 1583ee5..83ef83b 100644 --- a/src/styles/index.css +++ b/src/styles/index.css @@ -16,6 +16,8 @@ body, margin: 0; padding: 0; + overflow: hidden; + background-color: light-dark(var(--light-bg-color), var(--dark-bg-color)); } diff --git a/src/views/ConnectionInfoView.tsx b/src/views/ConnectionInfoView.tsx new file mode 100644 index 0000000..3507ec0 --- /dev/null +++ b/src/views/ConnectionInfoView.tsx @@ -0,0 +1,23 @@ +import type { ConnectionModel } from "@/models/connection"; +import { EditableText, H3, Label } from "@blueprintjs/core"; +import { observer } from "mobx-react"; + +interface ConnectionInfoViewProps { + connection: ConnectionModel; +} + +export const ConnectionInfoView = observer(({ connection }: ConnectionInfoViewProps) => { + return ( +
+

+ + { + connection.config.name = name; + }} + /> +

+
+ ); +}); diff --git a/src/views/ConnectionView.tsx b/src/views/ConnectionView.tsx deleted file mode 100644 index fa44d2c..0000000 --- a/src/views/ConnectionView.tsx +++ /dev/null @@ -1,149 +0,0 @@ -import { type BaseDiscUnion, matchUnions } from "@/lib/utils"; -import type { ConnectionModel } from "@/models/connection"; -import { Tree, type TreeNodeInfo } from "@blueprintjs/core"; -import { useQuery } from "@tanstack/react-query"; -import { observer, useLocalObservable } from "mobx-react"; -import { useState } from "react"; -import { SidebarLayout } from "../components/SidebarLayout"; -import { NewSavedQueryView } from "./NewSavedQueryView"; -import { TableBrowserView } from "./TableBrowserView"; -import { TableNodeView } from "./TableNodeView"; - -interface BaseView extends BaseDiscUnion {} - -interface DashboardView extends BaseView { - type: "dashboard"; -} - -interface TableView extends BaseView { - type: "table"; - schema: string; - table: string; -} - -interface GraphView extends BaseView { - type: "graph"; -} - -interface SavedQueryView extends BaseView { - type: "savedQuery"; -} - -type View = TableView | GraphView | DashboardView | SavedQueryView; - -interface SidebarProps { - connection: ConnectionModel; - view: View; - setView: (view: View) => void; -} - -interface TreeData { - onClick?: () => void; -} - -const Sidebar = observer(({ connection, view, setView }: SidebarProps) => { - const expanded = useLocalObservable(() => new Map()); - - const root: TreeNodeInfo[] = [ - { - id: "connection", - label: connection.name, - icon: "database", - isExpanded: expanded.get("connection") ?? true, - isSelected: view.type === "dashboard", - nodeData: { - onClick: () => setView({ type: "dashboard" }), - }, - - childNodes: [ - { - id: "visualize", - label: "Visualize", - icon: "graph", - isExpanded: expanded.get("visualize") ?? true, - isSelected: view.type === "graph", - nodeData: { - onClick: () => setView({ type: "graph" }), - }, - }, - { - id: "savedQueries", - label: "Queries", - icon: "antenna", - isExpanded: expanded.get("savedQueries") ?? true, - nodeData: { - onClick: () => setView({ type: "savedQuery" }), - }, - }, - { - id: "tables", - label: "Tables", - icon: "panel-table", - isExpanded: expanded.get("tables") ?? true, - - childNodes: connection.tables.map( - (table): TreeNodeInfo => ({ - id: `${table.name}.${table.name}`, - label: table.name, - nodeData: { - onClick: () => setView({ type: "table", schema: table.schema, table: table.name }), - }, - }), - ), - }, - ], - }, - ]; - - return ( - - onNodeClick={(node) => { - node.nodeData?.onClick?.(); - }} - onNodeCollapse={(node) => { - expanded.set(node.id as string, false); - }} - onNodeExpand={(node) => { - expanded.set(node.id as string, true); - }} - compact - contents={root} - /> - ); -}); - -interface ConnectionBrowserProps { - connection: ConnectionModel; -} - -export const ConnectionBrowser = observer(({ connection }: ConnectionBrowserProps) => { - const [selectedView, setSelectedView] = useState({ - type: "dashboard", - }); - useQuery({ - queryKey: ["updateMeta", connection.id], - queryFn: async () => { - return connection.updateMeta(); - }, - }); - - const SelectedView = matchUnions(selectedView, { - dashboard: () =>
Dashboard
, - graph: () => , - table: (v) => ( - - ), - savedQuery: () => , - }); - - return ( - }> - {SelectedView} - - ); -}); diff --git a/src/views/NewSavedQueryView.tsx b/src/views/SavedQueryView.tsx similarity index 58% rename from src/views/NewSavedQueryView.tsx rename to src/views/SavedQueryView.tsx index b25e9e3..183e65b 100644 --- a/src/views/NewSavedQueryView.tsx +++ b/src/views/SavedQueryView.tsx @@ -2,22 +2,75 @@ import type { ConnectionModel } from "@/models/connection"; import { Button, EditableText, H1, HTMLTable, Intent } from "@blueprintjs/core"; import { Editor } from "@monaco-editor/react"; import { observer, useLocalObservable } from "mobx-react"; -import { useState } from "react"; +import { useEffect, useMemo, useState } from "react"; import { formatDialect, postgresql } from "sql-formatter"; +import type { NewQuery, Query } from "@/lib/connection-config-file"; +import { useNavigate } from "@tanstack/react-router"; import { FaBroom } from "react-icons/fa"; -export interface NewSavedQueryViewProps { + +export interface SavedQueryViewProps { connection: ConnectionModel; + + queryId?: string; + + onSave: (query: Query | NewQuery) => Query; } -export const NewSavedQueryView = observer((props: NewSavedQueryViewProps) => { - const [name, setName] = useState(""); - const [query, setQuery] = useState("SELECT * FROM table WHERE id = $::value"); +export const SavedQueryEditorView = observer((props: SavedQueryViewProps) => { + const { connection, queryId, onSave } = props; + + const savedQuery = useMemo(() => { + return (connection.config.queries ?? []).find((q) => q.id === queryId); + }, [connection, queryId]); + + const [name, setName] = useState(savedQuery?.name ?? ""); + const [query, setQuery] = useState(savedQuery?.query ?? "SELECT * FROM table WHERE id = $::value"); const [newParamKey, setNewParamKey] = useState(""); const [newParamValue, setNewParamValue] = useState(""); - const params = useLocalObservable(() => new Map()); + const navigate = useNavigate(); + + const params = useLocalObservable(() => { + const params = new Map(); + + for (const param of savedQuery?.params ?? []) { + params.set(param.name, param.value); + } + + return params; + }); + + function reset() { + setName(savedQuery?.name ?? ""); + setQuery(savedQuery?.query ?? ""); + setNewParamKey(""); + setNewParamValue(""); + for (const param of savedQuery?.params ?? []) { + params.set(param.name, param.value); + } + } + + // biome-ignore lint/correctness/useExhaustiveDependencies: + useEffect(() => { + reset(); + }, [savedQuery]); + + const currQuery: NewQuery | Query = { + id: savedQuery?.id, + name, + description: "", + order: 0, + query, + params: Array.from(params.entries()).map(([name, value]) => ({ + name, + order: 0, + value, + type: "string", + })), + tagIds: [], + }; return (
@@ -91,8 +144,22 @@ export const NewSavedQueryView = observer((props: NewSavedQueryViewProps) => { />
-
); diff --git a/src/views/TagInfoView.tsx b/src/views/TagInfoView.tsx new file mode 100644 index 0000000..eaf99a0 --- /dev/null +++ b/src/views/TagInfoView.tsx @@ -0,0 +1,53 @@ +import type { TagInfo } from "@/lib/connection-config-file"; +import type { ConnectionModel } from "@/models/connection"; +import { EditableText, H1 } from "@blueprintjs/core"; +import { runInAction } from "mobx"; +import { observer } from "mobx-react"; + +interface TagInfoViewProps { + tagId: string; + connection: ConnectionModel; +} + +export const TagInfoView = observer((props: TagInfoViewProps) => { + const { tagId, connection } = props; + + const entry = connection.config.tags.find((t) => t.id === tagId); + + if (!entry) { + return ( +
+

Tag not found

+
+ ); + } + + return ; +}); + +interface TagInfoEntryViewProps { + entry: TagInfo; + connection: ConnectionModel; +} + +export const TagInfoEntryView = observer((props: TagInfoEntryViewProps) => { + const { entry } = props; + + const name = entry.name; + + return ( +
+

+ { + runInAction(() => { + entry.name = name; + }); + }} + /> +

+ Tag: +
+ ); +}); diff --git a/vite.config.ts b/vite.config.ts index 4abe53a..dd26b21 100644 --- a/vite.config.ts +++ b/vite.config.ts @@ -3,8 +3,10 @@ import react from "@vitejs/plugin-react"; import { defineConfig } from "vite"; import tsconfigPaths from "vite-tsconfig-paths"; +const host = process.env.TAURI_DEV_HOST; + // https://vitejs.dev/config/ -export default defineConfig(async () => ({ +export default defineConfig({ plugins: [TanStackRouterVite(), react(), tsconfigPaths()], esbuild: { target: "ES2022", @@ -13,17 +15,26 @@ export default defineConfig(async () => ({ }, }, + build: { + // Tauri uses Chromium on Windows and WebKit on macOS and Linux + target: process.env.TAURI_ENV_PLATFORM === "windows" ? "chrome105" : "safari13", + // don't minify for debug builds + minify: !process.env.TAURI_ENV_DEBUG ? "esbuild" : false, + // produce sourcemaps for debug builds + sourcemap: !!process.env.TAURI_ENV_DEBUG, + }, + + envPrefix: ["VITE_", "TAURI_ENV_*"], + // Vite options tailored for Tauri development and only applied in `tauri dev` or `tauri build` // // 1. prevent vite from obscuring rust errors clearScreen: false, - // 2. tauri expects a fixed port, fail if that port is not available server: { - port: 1420, + // Tauri expects a fixed port, fail if that port is not available strictPort: true, - watch: { - // 3. tell vite to ignore watching `src-tauri` - ignored: ["**/src-tauri/**"], - }, + // if the host Tauri is expecting is set, use it + host: true, + port: 5173, }, -})); +});