feat: implement wally support

This commit is contained in:
daimond113 2024-08-08 17:59:59 +02:00
parent a24a440a84
commit a8a8ffcbe2
No known key found for this signature in database
GPG key ID: 3A8ECE51328B513C
32 changed files with 1466 additions and 594 deletions

222
Cargo.lock generated
View file

@ -160,16 +160,16 @@ dependencies = [
[[package]] [[package]]
name = "actix-server" name = "actix-server"
version = "2.4.0" version = "2.5.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b02303ce8d4e8be5b855af6cf3c3a08f3eff26880faad82bab679c22d3650cb5" checksum = "7ca2549781d8dd6d75c40cf6b6051260a2cc2f3c62343d761a969a0640646894"
dependencies = [ dependencies = [
"actix-rt", "actix-rt",
"actix-service", "actix-service",
"actix-utils", "actix-utils",
"futures-core", "futures-core",
"futures-util", "futures-util",
"mio 0.8.11", "mio 1.0.1",
"socket2", "socket2",
"tokio", "tokio",
"tracing", "tracing",
@ -251,9 +251,9 @@ dependencies = [
[[package]] [[package]]
name = "actix-web-lab" name = "actix-web-lab"
version = "0.20.2" version = "0.21.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "7675c1a84eec1b179c844cdea8488e3e409d8e4984026e92fa96c87dd86f33c6" checksum = "993cb5477d926300f11d7daede86368dbf797ccae3564aadfb41cb5b3aee2d23"
dependencies = [ dependencies = [
"actix-http", "actix-http",
"actix-router", "actix-router",
@ -263,7 +263,6 @@ dependencies = [
"actix-web-lab-derive", "actix-web-lab-derive",
"ahash", "ahash",
"arc-swap", "arc-swap",
"async-trait",
"bytes", "bytes",
"bytestring", "bytestring",
"csv", "csv",
@ -272,7 +271,7 @@ dependencies = [
"futures-util", "futures-util",
"http 0.2.12", "http 0.2.12",
"impl-more", "impl-more",
"itertools", "itertools 0.13.0",
"local-channel", "local-channel",
"mediatype", "mediatype",
"mime", "mime",
@ -289,9 +288,9 @@ dependencies = [
[[package]] [[package]]
name = "actix-web-lab-derive" name = "actix-web-lab-derive"
version = "0.20.0" version = "0.21.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9aa0b287c8de4a76b691f29dbb5451e8dd5b79d777eaf87350c9b0cbfdb5e968" checksum = "5aa1bc8506ff10e35419d82d2502e182b94bafa1a68f5651e8e1e6c6717fe1d3"
dependencies = [ dependencies = [
"proc-macro2", "proc-macro2",
"quote", "quote",
@ -715,9 +714,9 @@ checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b"
[[package]] [[package]]
name = "bytes" name = "bytes"
version = "1.6.1" version = "1.7.1"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a12916984aab3fa6e39d655a33e09c0071eb36d6ab3aea5c2d78551f1df6d952" checksum = "8318a53db07bb3f8dca91a600466bdb3f2eaadeedfdbcf02e1accbad9271ba50"
[[package]] [[package]]
name = "bytestring" name = "bytestring"
@ -760,9 +759,9 @@ dependencies = [
[[package]] [[package]]
name = "cc" name = "cc"
version = "1.1.6" version = "1.1.8"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2aba8f4e9906c7ce3c73463f62a7f0c65183ada1a2d47e397cc8810827f9694f" checksum = "504bdec147f2cc13c8b57ed9401fd8a147cc66b67ad5cb241394244f2c947549"
dependencies = [ dependencies = [
"jobserver", "jobserver",
"libc", "libc",
@ -813,9 +812,9 @@ dependencies = [
[[package]] [[package]]
name = "clap" name = "clap"
version = "4.5.11" version = "4.5.13"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "35723e6a11662c2afb578bcf0b88bf6ea8e21282a953428f240574fcc3a2b5b3" checksum = "0fbb260a053428790f3de475e304ff84cdbc4face759ea7a3e64c1edd938a7fc"
dependencies = [ dependencies = [
"clap_builder", "clap_builder",
"clap_derive", "clap_derive",
@ -823,9 +822,9 @@ dependencies = [
[[package]] [[package]]
name = "clap_builder" name = "clap_builder"
version = "4.5.11" version = "4.5.13"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "49eb96cbfa7cfa35017b7cd548c75b14c3118c98b423041d70562665e07fb0fa" checksum = "64b17d7ea74e9f833c7dbf2cbe4fb12ff26783eda4782a8975b72f895c9b4d99"
dependencies = [ dependencies = [
"anstream", "anstream",
"anstyle", "anstyle",
@ -835,9 +834,9 @@ dependencies = [
[[package]] [[package]]
name = "clap_derive" name = "clap_derive"
version = "4.5.11" version = "4.5.13"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "5d029b67f89d30bbb547c89fd5161293c0aec155fc691d7924b64550662db93e" checksum = "501d359d5f3dcaf6ecdeee48833ae73ec6e42723a1e52419c79abf9507eec0a0"
dependencies = [ dependencies = [
"heck", "heck",
"proc-macro2", "proc-macro2",
@ -1247,9 +1246,9 @@ checksum = "75b325c5dbd37f80359721ad39aca5a29fb04c89279657cffdda8736d0c0b9d2"
[[package]] [[package]]
name = "dunce" name = "dunce"
version = "1.0.4" version = "1.0.5"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "56ce8c6da7551ec6c462cbaf3bfbc75131ebbfa1c944aeaa9dab51ca1c5f0c3b" checksum = "92773504d58c093f6de2459af4af33faa518c13451eb8f2b5698ed3d36e7c813"
[[package]] [[package]]
name = "dyn-clone" name = "dyn-clone"
@ -1399,9 +1398,9 @@ dependencies = [
[[package]] [[package]]
name = "flate2" name = "flate2"
version = "1.0.30" version = "1.0.31"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "5f54427cfd1c7829e2a139fcefea601bf088ebca651d2bf53ebc600eac295dae" checksum = "7f211bbe8e69bbd0cfdea405084f128ae8b4aaa6b0b522fc8f2b009084797920"
dependencies = [ dependencies = [
"crc32fast", "crc32fast",
"miniz_oxide", "miniz_oxide",
@ -1707,7 +1706,7 @@ dependencies = [
"gix-utils", "gix-utils",
"itoa", "itoa",
"thiserror", "thiserror",
"winnow 0.6.16", "winnow 0.6.18",
] ]
[[package]] [[package]]
@ -1789,7 +1788,7 @@ dependencies = [
"smallvec", "smallvec",
"thiserror", "thiserror",
"unicode-bom", "unicode-bom",
"winnow 0.6.16", "winnow 0.6.18",
] ]
[[package]] [[package]]
@ -2042,7 +2041,7 @@ dependencies = [
"itoa", "itoa",
"smallvec", "smallvec",
"thiserror", "thiserror",
"winnow 0.6.16", "winnow 0.6.18",
] ]
[[package]] [[package]]
@ -2165,7 +2164,7 @@ dependencies = [
"gix-utils", "gix-utils",
"maybe-async", "maybe-async",
"thiserror", "thiserror",
"winnow 0.6.16", "winnow 0.6.18",
] ]
[[package]] [[package]]
@ -2197,7 +2196,7 @@ dependencies = [
"gix-validate", "gix-validate",
"memmap2", "memmap2",
"thiserror", "thiserror",
"winnow 0.6.16", "winnow 0.6.18",
] ]
[[package]] [[package]]
@ -2432,7 +2431,7 @@ dependencies = [
"futures-sink", "futures-sink",
"futures-util", "futures-util",
"http 0.2.12", "http 0.2.12",
"indexmap 2.2.6", "indexmap 2.3.0",
"slab", "slab",
"tokio", "tokio",
"tokio-util", "tokio-util",
@ -2451,7 +2450,7 @@ dependencies = [
"futures-core", "futures-core",
"futures-sink", "futures-sink",
"http 1.1.0", "http 1.1.0",
"indexmap 2.2.6", "indexmap 2.3.0",
"slab", "slab",
"tokio", "tokio",
"tokio-util", "tokio-util",
@ -2661,9 +2660,9 @@ dependencies = [
[[package]] [[package]]
name = "hyper-util" name = "hyper-util"
version = "0.1.6" version = "0.1.7"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3ab92f4f49ee4fb4f997c784b7a2e0fa70050211e0b6a287f898c3c9785ca956" checksum = "cde7055719c54e36e95e8719f95883f22072a48ede39db7fc17a4e1d5281e9b9"
dependencies = [ dependencies = [
"bytes", "bytes",
"futures-channel", "futures-channel",
@ -2737,9 +2736,9 @@ dependencies = [
[[package]] [[package]]
name = "indexmap" name = "indexmap"
version = "2.2.6" version = "2.3.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "168fb715dda47215e360912c096649d23d58bf392ac62f73919e831745e40f26" checksum = "de3fc2e30ba82dd1b3911c8de1ffc143c74a914a14e99514d7637e3099df5ea0"
dependencies = [ dependencies = [
"equivalent", "equivalent",
"hashbrown 0.14.5", "hashbrown 0.14.5",
@ -2869,6 +2868,15 @@ dependencies = [
"either", "either",
] ]
[[package]]
name = "itertools"
version = "0.13.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "413ee7dfc52ee1a4949ceeb7dbc8a33f2d6c088194d9f922fb8318faf1f01186"
dependencies = [
"either",
]
[[package]] [[package]]
name = "itoa" name = "itoa"
version = "1.0.11" version = "1.0.11"
@ -2895,9 +2903,9 @@ dependencies = [
[[package]] [[package]]
name = "keyring" name = "keyring"
version = "3.0.4" version = "3.0.5"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "c118b1bc529b034aad851808f41f49a69a337d10e112039e7f342e5fd514635b" checksum = "0c163ef0b9da5ccf44ae4d7c9d24fb1a8750aa1969d484865fc1eedc44b26c09"
dependencies = [ dependencies = [
"byteorder", "byteorder",
"dbus-secret-service", "dbus-secret-service",
@ -3062,9 +3070,9 @@ checksum = "a7a70ba024b9dc04c27ea2f0c0548feb474ec5c54bba33a7f72f873a39d07b24"
[[package]] [[package]]
name = "lru" name = "lru"
version = "0.12.3" version = "0.12.4"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d3262e75e648fce39813cb56ac41f3c3e3f65217ebf3844d818d1f9398cfb0dc" checksum = "37ee39891760e7d94734f6f63fedc29a2e4a152f836120753a72503f09fcf904"
dependencies = [ dependencies = [
"hashbrown 0.14.5", "hashbrown 0.14.5",
] ]
@ -3197,6 +3205,7 @@ checksum = "4569e456d394deccd22ce1c1913e6ea0e54519f577285001215d33557431afe4"
dependencies = [ dependencies = [
"hermit-abi 0.3.9", "hermit-abi 0.3.9",
"libc", "libc",
"log",
"wasi", "wasi",
"windows-sys 0.52.0", "windows-sys 0.52.0",
] ]
@ -3375,9 +3384,9 @@ checksum = "830b246a0e5f20af87141b25c173cd1b609bd7779a4617d6ec582abaf90870f3"
[[package]] [[package]]
name = "object" name = "object"
version = "0.36.2" version = "0.36.3"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3f203fa8daa7bb185f760ae12bd8e097f63d17041dcdcaf675ac54cdf863170e" checksum = "27b64972346851a39438c60b341ebc01bba47464ae329e55cf343eb93964efd9"
dependencies = [ dependencies = [
"memchr", "memchr",
] ]
@ -3577,10 +3586,11 @@ dependencies = [
"serde_with", "serde_with",
"sha2", "sha2",
"tar", "tar",
"tempfile",
"thiserror", "thiserror",
"threadpool", "threadpool",
"toml", "toml",
"toml_edit 0.22.17", "toml_edit 0.22.20",
"url", "url",
"winreg", "winreg",
"zip", "zip",
@ -3695,9 +3705,12 @@ checksum = "439ee305def115ba05938db6eb1644ff94165c5ab5e9420d1c1bcedbba909391"
[[package]] [[package]]
name = "ppv-lite86" name = "ppv-lite86"
version = "0.2.17" version = "0.2.20"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "5b40af805b3121feab8a3c29f04d8ad262fa8e0561883e7653e024ae4479e6de" checksum = "77957b295656769bb8ad2b6a6b09d897d94f05c41b069aede1fcdaa675eaea04"
dependencies = [
"zerocopy",
]
[[package]] [[package]]
name = "pretty_env_logger" name = "pretty_env_logger"
@ -3760,16 +3773,17 @@ dependencies = [
[[package]] [[package]]
name = "quinn" name = "quinn"
version = "0.11.2" version = "0.11.3"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e4ceeeeabace7857413798eb1ffa1e9c905a9946a57d81fb69b4b71c4d8eb3ad" checksum = "b22d8e7369034b9a7132bc2008cac12f2013c8132b45e0554e6e20e2617f2156"
dependencies = [ dependencies = [
"bytes", "bytes",
"pin-project-lite", "pin-project-lite",
"quinn-proto", "quinn-proto",
"quinn-udp", "quinn-udp",
"rustc-hash", "rustc-hash 2.0.0",
"rustls", "rustls",
"socket2",
"thiserror", "thiserror",
"tokio", "tokio",
"tracing", "tracing",
@ -3777,14 +3791,14 @@ dependencies = [
[[package]] [[package]]
name = "quinn-proto" name = "quinn-proto"
version = "0.11.3" version = "0.11.6"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ddf517c03a109db8100448a4be38d498df8a210a99fe0e1b9eaf39e78c640efe" checksum = "ba92fb39ec7ad06ca2582c0ca834dfeadcaf06ddfc8e635c80aa7e1c05315fdd"
dependencies = [ dependencies = [
"bytes", "bytes",
"rand", "rand",
"ring", "ring",
"rustc-hash", "rustc-hash 2.0.0",
"rustls", "rustls",
"slab", "slab",
"thiserror", "thiserror",
@ -3801,6 +3815,7 @@ dependencies = [
"libc", "libc",
"once_cell", "once_cell",
"socket2", "socket2",
"tracing",
"windows-sys 0.52.0", "windows-sys 0.52.0",
] ]
@ -3913,9 +3928,9 @@ dependencies = [
[[package]] [[package]]
name = "regex" name = "regex"
version = "1.10.5" version = "1.10.6"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b91213439dad192326a0d7c6ee3955910425f441d7038e0d6933b0aec5c4517f" checksum = "4219d74c6b67a3654a9fbebc4b419e22126d13d2f3c4a07ee0cb61ff79a79619"
dependencies = [ dependencies = [
"aho-corasick", "aho-corasick",
"memchr", "memchr",
@ -4042,6 +4057,12 @@ version = "1.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "08d43f7aa6b08d49f382cde6a7982047c3426db949b1424bc4b7ec9ae12c6ce2" checksum = "08d43f7aa6b08d49f382cde6a7982047c3426db949b1424bc4b7ec9ae12c6ce2"
[[package]]
name = "rustc-hash"
version = "2.0.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "583034fd73374156e66797ed8e5b0d5690409c9226b22d87cb7f19821c05d152"
[[package]] [[package]]
name = "rustc_version" name = "rustc_version"
version = "0.4.0" version = "0.4.0"
@ -4080,9 +4101,9 @@ dependencies = [
[[package]] [[package]]
name = "rustls-pemfile" name = "rustls-pemfile"
version = "2.1.2" version = "2.1.3"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "29993a25686778eb88d4189742cd713c9bce943bc54251a33509dc63cbacf73d" checksum = "196fe16b00e106300d3e45ecfcb764fa292a535d7326a29a5875c579c7417425"
dependencies = [ dependencies = [
"base64 0.22.1", "base64 0.22.1",
"rustls-pki-types", "rustls-pki-types",
@ -4090,9 +4111,9 @@ dependencies = [
[[package]] [[package]]
name = "rustls-pki-types" name = "rustls-pki-types"
version = "1.7.0" version = "1.8.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "976295e77ce332211c0d24d92c0e83e50f5c5f046d11082cea19f3df13a3562d" checksum = "fc0a2ce646f8655401bb81e7927b812614bd5d91dbc968696be50603510fcaf0"
[[package]] [[package]]
name = "rustls-webpki" name = "rustls-webpki"
@ -4336,18 +4357,18 @@ dependencies = [
[[package]] [[package]]
name = "serde" name = "serde"
version = "1.0.204" version = "1.0.205"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "bc76f558e0cbb2a839d37354c575f1dc3fdc6546b5be373ba43d95f231bf7c12" checksum = "e33aedb1a7135da52b7c21791455563facbbcc43d0f0f66165b42c21b3dfb150"
dependencies = [ dependencies = [
"serde_derive", "serde_derive",
] ]
[[package]] [[package]]
name = "serde_derive" name = "serde_derive"
version = "1.0.204" version = "1.0.205"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e0cd7e117be63d3c3678776753929474f3b04a43a080c744d6b0ae2a8c28e222" checksum = "692d6f5ac90220161d6774db30c662202721e64aed9058d2c394f451261420c1"
dependencies = [ dependencies = [
"proc-macro2", "proc-macro2",
"quote", "quote",
@ -4361,7 +4382,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8de514ef58196f1fc96dcaef80fe6170a1ce6215df9687a93fe8300e773fefc5" checksum = "8de514ef58196f1fc96dcaef80fe6170a1ce6215df9687a93fe8300e773fefc5"
dependencies = [ dependencies = [
"form_urlencoded", "form_urlencoded",
"indexmap 2.2.6", "indexmap 2.3.0",
"itoa", "itoa",
"ryu", "ryu",
"serde", "serde",
@ -4369,9 +4390,9 @@ dependencies = [
[[package]] [[package]]
name = "serde_json" name = "serde_json"
version = "1.0.121" version = "1.0.122"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "4ab380d7d9f22ef3f21ad3e6c1ebe8e4fc7a2000ccba2e4d71fc96f15b2cb609" checksum = "784b6203951c57ff748476b126ccb5e8e2959a5c19e5c617ab1956be3dbc68da"
dependencies = [ dependencies = [
"itoa", "itoa",
"memchr", "memchr",
@ -4430,7 +4451,7 @@ dependencies = [
"chrono", "chrono",
"hex", "hex",
"indexmap 1.9.3", "indexmap 1.9.3",
"indexmap 2.2.6", "indexmap 2.3.0",
"serde", "serde",
"serde_derive", "serde_derive",
"serde_json", "serde_json",
@ -4496,9 +4517,9 @@ dependencies = [
[[package]] [[package]]
name = "signal-hook-mio" name = "signal-hook-mio"
version = "0.2.3" version = "0.2.4"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "29ad2e15f37ec9a6cc544097b78a1ec90001e9f71b81338ca39f430adaca99af" checksum = "34db1a06d485c9142248b7a054f034b349b212551f3dfd19c94d45a754a217cd"
dependencies = [ dependencies = [
"libc", "libc",
"mio 0.8.11", "mio 0.8.11",
@ -4670,7 +4691,7 @@ dependencies = [
"fnv", "fnv",
"fs4", "fs4",
"htmlescape", "htmlescape",
"itertools", "itertools 0.12.1",
"levenshtein_automata", "levenshtein_automata",
"log", "log",
"lru", "lru",
@ -4683,7 +4704,7 @@ dependencies = [
"rayon", "rayon",
"regex", "regex",
"rust-stemmers", "rust-stemmers",
"rustc-hash", "rustc-hash 1.1.0",
"serde", "serde",
"serde_json", "serde_json",
"sketches-ddsketch", "sketches-ddsketch",
@ -4719,7 +4740,7 @@ checksum = "12722224ffbe346c7fec3275c699e508fd0d4710e629e933d5736ec524a1f44e"
dependencies = [ dependencies = [
"downcast-rs", "downcast-rs",
"fastdivide", "fastdivide",
"itertools", "itertools 0.12.1",
"serde", "serde",
"tantivy-bitpacker", "tantivy-bitpacker",
"tantivy-common", "tantivy-common",
@ -4805,14 +4826,15 @@ dependencies = [
[[package]] [[package]]
name = "tempfile" name = "tempfile"
version = "3.10.1" version = "3.12.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "85b77fafb263dd9d05cbeac119526425676db3784113aa9295c88498cbf8bff1" checksum = "04cbcdd0c794ebb0d4cf35e88edd2f7d2c4c3e9a5a6dab322839b321c6a87a64"
dependencies = [ dependencies = [
"cfg-if", "cfg-if",
"fastrand", "fastrand",
"once_cell",
"rustix", "rustix",
"windows-sys 0.52.0", "windows-sys 0.59.0",
] ]
[[package]] [[package]]
@ -4913,9 +4935,9 @@ checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20"
[[package]] [[package]]
name = "tokio" name = "tokio"
version = "1.39.1" version = "1.39.2"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d040ac2b29ab03b09d4129c2f5bbd012a3ac2f79d38ff506a4bf8dd34b0eac8a" checksum = "daa4fb1bc778bd6f04cbfc4bb2d06a7396a8f299dc33ea1900cedaa316f467b1"
dependencies = [ dependencies = [
"backtrace", "backtrace",
"bytes", "bytes",
@ -4987,21 +5009,21 @@ dependencies = [
[[package]] [[package]]
name = "toml" name = "toml"
version = "0.8.16" version = "0.8.19"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "81967dd0dd2c1ab0bc3468bd7caecc32b8a4aa47d0c8c695d8c2b2108168d62c" checksum = "a1ed1f98e3fdc28d6d910e6737ae6ab1a93bf1985935a1193e68f93eeb68d24e"
dependencies = [ dependencies = [
"serde", "serde",
"serde_spanned", "serde_spanned",
"toml_datetime", "toml_datetime",
"toml_edit 0.22.17", "toml_edit 0.22.20",
] ]
[[package]] [[package]]
name = "toml_datetime" name = "toml_datetime"
version = "0.6.7" version = "0.6.8"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f8fb9f64314842840f1d940ac544da178732128f1c78c21772e876579e0da1db" checksum = "0dd7358ecb8fc2f8d014bf86f6f638ce72ba252a2c3a2572f2a795f1d23efb41"
dependencies = [ dependencies = [
"serde", "serde",
] ]
@ -5012,22 +5034,22 @@ version = "0.21.1"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "6a8534fd7f78b5405e860340ad6575217ce99f38d4d5c8f2442cb5ecb50090e1" checksum = "6a8534fd7f78b5405e860340ad6575217ce99f38d4d5c8f2442cb5ecb50090e1"
dependencies = [ dependencies = [
"indexmap 2.2.6", "indexmap 2.3.0",
"toml_datetime", "toml_datetime",
"winnow 0.5.40", "winnow 0.5.40",
] ]
[[package]] [[package]]
name = "toml_edit" name = "toml_edit"
version = "0.22.17" version = "0.22.20"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8d9f8729f5aea9562aac1cc0441f5d6de3cff1ee0c5d67293eeca5eb36ee7c16" checksum = "583c44c02ad26b0c3f3066fe629275e50627026c51ac2e595cca4c230ce1ce1d"
dependencies = [ dependencies = [
"indexmap 2.2.6", "indexmap 2.3.0",
"serde", "serde",
"serde_spanned", "serde_spanned",
"toml_datetime", "toml_datetime",
"winnow 0.6.16", "winnow 0.6.18",
] ]
[[package]] [[package]]
@ -5378,11 +5400,11 @@ checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6"
[[package]] [[package]]
name = "winapi-util" name = "winapi-util"
version = "0.1.8" version = "0.1.9"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "4d4cc384e1e73b93bafa6fb4f1df8c41695c8a91cf9c4c64358067d15a7b6c6b" checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb"
dependencies = [ dependencies = [
"windows-sys 0.52.0", "windows-sys 0.59.0",
] ]
[[package]] [[package]]
@ -5428,6 +5450,15 @@ dependencies = [
"windows-targets 0.52.6", "windows-targets 0.52.6",
] ]
[[package]]
name = "windows-sys"
version = "0.59.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1e38bc4d79ed67fd075bcc251a1c39b32a1776bbe92e5bef1f0bf1f8c531853b"
dependencies = [
"windows-targets 0.52.6",
]
[[package]] [[package]]
name = "windows-targets" name = "windows-targets"
version = "0.48.5" version = "0.48.5"
@ -5560,9 +5591,9 @@ dependencies = [
[[package]] [[package]]
name = "winnow" name = "winnow"
version = "0.6.16" version = "0.6.18"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b480ae9340fc261e6be3e95a1ba86d54ae3f9171132a73ce8d4bbaf68339507c" checksum = "68a9bda4691f099d435ad181000724da8e5899daa10713c2d432552b9ccd3a6f"
dependencies = [ dependencies = [
"memchr", "memchr",
] ]
@ -5660,6 +5691,7 @@ version = "0.7.35"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "1b9b4fd18abc82b8136838da5d50bae7bdea537c574d8dc1a34ed098d6c166f0" checksum = "1b9b4fd18abc82b8136838da5d50bae7bdea537c574d8dc1a34ed098d6c166f0"
dependencies = [ dependencies = [
"byteorder",
"zerocopy-derive", "zerocopy-derive",
] ]
@ -5696,9 +5728,9 @@ dependencies = [
[[package]] [[package]]
name = "zip" name = "zip"
version = "2.1.5" version = "2.1.6"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b895748a3ebcb69b9d38dcfdf21760859a4b0d0b0015277640c2ef4c69640e6f" checksum = "40dd8c92efc296286ce1fbd16657c5dbefff44f1b4ca01cc5f517d8b7b3d3e2e"
dependencies = [ dependencies = [
"aes", "aes",
"arbitrary", "arbitrary",
@ -5710,7 +5742,7 @@ dependencies = [
"displaydoc", "displaydoc",
"flate2", "flate2",
"hmac", "hmac",
"indexmap 2.2.6", "indexmap 2.3.0",
"lzma-rs", "lzma-rs",
"memchr", "memchr",
"pbkdf2", "pbkdf2",
@ -5748,18 +5780,18 @@ dependencies = [
[[package]] [[package]]
name = "zstd-safe" name = "zstd-safe"
version = "7.2.0" version = "7.2.1"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "fa556e971e7b568dc775c136fc9de8c779b1c2fc3a63defaafadffdbd3181afa" checksum = "54a3ab4db68cea366acc5c897c7b4d4d1b8994a9cd6e6f841f8964566a419059"
dependencies = [ dependencies = [
"zstd-sys", "zstd-sys",
] ]
[[package]] [[package]]
name = "zstd-sys" name = "zstd-sys"
version = "2.0.12+zstd.1.5.6" version = "2.0.13+zstd.1.5.6"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "0a4e40c320c3cb459d9a9ff6de98cff88f4751ee9275d140e2be94a2b74e4c13" checksum = "38ff0f21cfee8f97d94cef41359e0c89aa6113028ab0291aa8ca0038995a95aa"
dependencies = [ dependencies = [
"cc", "cc",
"pkg-config", "pkg-config",

View file

@ -44,13 +44,13 @@ uninlined_format_args = "warn"
[dependencies] [dependencies]
serde = { version = "1.0.204", features = ["derive"] } serde = { version = "1.0.204", features = ["derive"] }
toml = "0.8.16" toml = "0.8.19"
serde_with = "3.9.0" serde_with = "3.9.0"
gix = { version = "0.64.0", default-features = false, features = ["blocking-http-transport-reqwest-rust-tls", "revparse-regex", "credentials"] } gix = { version = "0.64.0", default-features = false, features = ["blocking-http-transport-reqwest-rust-tls", "revparse-regex", "credentials"] }
semver = { version = "1.0.23", features = ["serde"] } semver = { version = "1.0.23", features = ["serde"] }
reqwest = { version = "0.12.5", default-features = false, features = ["rustls-tls", "blocking"] } reqwest = { version = "0.12.5", default-features = false, features = ["rustls-tls", "blocking"] }
tar = "0.4.41" tar = "0.4.41"
flate2 = "1.0.30" flate2 = "1.0.31"
pathdiff = "0.2.1" pathdiff = "0.2.1"
relative-path = { version = "1.9.3", features = ["serde"] } relative-path = { version = "1.9.3", features = ["serde"] }
log = "0.4.22" log = "0.4.22"
@ -62,19 +62,20 @@ url = { version = "2.5.2", features = ["serde"] }
# secrecy = "0.8.0" # secrecy = "0.8.0"
chrono = { version = "0.4.38", features = ["serde"] } chrono = { version = "0.4.38", features = ["serde"] }
sha2 = "0.10.8" sha2 = "0.10.8"
tempfile = "3.12.0"
# TODO: remove this when gitoxide adds support for: committing, pushing, adding # TODO: remove this when gitoxide adds support for: committing, pushing, adding
git2 = { version = "0.19.0", optional = true } git2 = { version = "0.19.0", optional = true }
zip = { version = "2.1.5", optional = true } zip = { version = "2.1.6", optional = true }
serde_json = { version = "1.0.121", optional = true } serde_json = { version = "1.0.122", optional = true }
anyhow = { version = "1.0.86", optional = true } anyhow = { version = "1.0.86", optional = true }
open = { version = "5.3.0", optional = true } open = { version = "5.3.0", optional = true }
keyring = { version = "3.0.4", features = ["crypto-rust", "windows-native", "apple-native", "linux-native"], optional = true } keyring = { version = "3.0.5", features = ["crypto-rust", "windows-native", "apple-native", "linux-native"], optional = true }
colored = { version = "2.1.0", optional = true } colored = { version = "2.1.0", optional = true }
toml_edit = { version = "0.22.17", optional = true } toml_edit = { version = "0.22.20", optional = true }
clap = { version = "4.5.11", features = ["derive"], optional = true } clap = { version = "4.5.13", features = ["derive"], optional = true }
dirs = { version = "5.0.1", optional = true } dirs = { version = "5.0.1", optional = true }
pretty_env_logger = { version = "0.5.0", optional = true } pretty_env_logger = { version = "0.5.0", optional = true }
indicatif = { version = "0.17.8", optional = true } indicatif = { version = "0.17.8", optional = true }

View file

@ -7,8 +7,8 @@ publish = false
[dependencies] [dependencies]
actix-web = "4.8.0" actix-web = "4.8.0"
actix-web-lab = "0.20.2" actix-web-lab = "0.21.0"
actix-multipart = { version = "0.7.2", features = ["derive"] } actix-multipart = "0.7.2"
actix-cors = "0.7.0" actix-cors = "0.7.0"
actix-governor = "0.5.0" actix-governor = "0.5.0"
dotenvy = "0.15.7" dotenvy = "0.15.7"

View file

@ -5,13 +5,16 @@ use rusty_s3::{actions::GetObject, S3Action};
use semver::Version; use semver::Version;
use serde::{Deserialize, Deserializer}; use serde::{Deserialize, Deserializer};
use pesde::{manifest::target::TargetKind, names::PackageName, source::pesde::IndexFile};
use crate::{ use crate::{
error::Error, error::Error,
package::{s3_name, PackageResponse, S3_SIGN_DURATION}, package::{s3_name, PackageResponse, S3_SIGN_DURATION},
AppState, AppState,
}; };
use pesde::{
manifest::target::TargetKind,
names::PackageName,
source::{git_index::GitBasedSource, pesde::IndexFile},
};
#[derive(Debug)] #[derive(Debug)]
pub enum VersionRequest { pub enum VersionRequest {

View file

@ -1,8 +1,10 @@
use actix_web::{web, HttpResponse, Responder}; use actix_web::{web, HttpResponse, Responder};
use pesde::{names::PackageName, source::pesde::IndexFile};
use crate::{error::Error, package::PackageResponse, AppState}; use crate::{error::Error, package::PackageResponse, AppState};
use pesde::{
names::PackageName,
source::{git_index::GitBasedSource, pesde::IndexFile},
};
pub async fn get_package_versions( pub async fn get_package_versions(
app_state: web::Data<AppState>, app_state: web::Data<AppState>,

View file

@ -3,8 +3,9 @@ use std::{
io::{Cursor, Read, Write}, io::{Cursor, Read, Write},
}; };
use actix_multipart::form::{bytes::Bytes, MultipartForm}; use actix_multipart::Multipart;
use actix_web::{web, HttpResponse, Responder}; use actix_web::{web, HttpResponse, Responder};
use actix_web_lab::__reexports::futures_util::StreamExt;
use flate2::read::GzDecoder; use flate2::read::GzDecoder;
use git2::{Remote, Repository, Signature}; use git2::{Remote, Repository, Signature};
use rusty_s3::{actions::PutObject, S3Action}; use rusty_s3::{actions::PutObject, S3Action};
@ -13,9 +14,9 @@ use tar::Archive;
use pesde::{ use pesde::{
manifest::Manifest, manifest::Manifest,
source::{ source::{
git_index::GitBasedSource,
pesde::{IndexFile, IndexFileEntry, ScopeInfo, SCOPE_INFO_FILE}, pesde::{IndexFile, IndexFileEntry, ScopeInfo, SCOPE_INFO_FILE},
specifiers::DependencySpecifiers, specifiers::DependencySpecifiers,
traits::PackageSource,
version_id::VersionId, version_id::VersionId,
}, },
DEFAULT_INDEX_NAME, MANIFEST_FILE_NAME, DEFAULT_INDEX_NAME, MANIFEST_FILE_NAME,
@ -30,12 +31,6 @@ use crate::{
AppState, AppState,
}; };
#[derive(MultipartForm)]
pub struct PublishBody {
#[multipart(limit = "4 MiB")]
tarball: Bytes,
}
fn signature<'a>() -> Signature<'a> { fn signature<'a>() -> Signature<'a> {
Signature::now( Signature::now(
&benv!(required "COMMITTER_GIT_NAME"), &benv!(required "COMMITTER_GIT_NAME"),
@ -63,10 +58,24 @@ const FORBIDDEN_DIRECTORIES: &[&str] = &[".git"];
pub async fn publish_package( pub async fn publish_package(
app_state: web::Data<AppState>, app_state: web::Data<AppState>,
body: MultipartForm<PublishBody>, mut body: Multipart,
user_id: web::ReqData<UserId>, user_id: web::ReqData<UserId>,
) -> Result<impl Responder, Error> { ) -> Result<impl Responder, Error> {
let bytes = body.tarball.data.to_vec(); let max_archive_size = {
let source = app_state.source.lock().unwrap();
source.refresh(&app_state.project).map_err(Box::new)?;
source.config(&app_state.project)?.max_archive_size
};
let bytes = body
.next()
.await
.ok_or(Error::InvalidArchive)?
.map_err(|_| Error::InvalidArchive)?
.bytes(max_archive_size)
.await
.map_err(|_| Error::InvalidArchive)?
.map_err(|_| Error::InvalidArchive)?;
let mut decoder = GzDecoder::new(Cursor::new(&bytes)); let mut decoder = GzDecoder::new(Cursor::new(&bytes));
let mut archive = Archive::new(&mut decoder); let mut archive = Archive::new(&mut decoder);
@ -139,6 +148,11 @@ pub async fn publish_package(
return Err(Error::InvalidArchive); return Err(Error::InvalidArchive);
} }
} }
DependencySpecifiers::Wally(_) => {
if !config.wally_allowed {
return Err(Error::InvalidArchive);
}
}
}; };
} }

View file

@ -4,9 +4,11 @@ use actix_web::{web, HttpResponse, Responder};
use serde::Deserialize; use serde::Deserialize;
use tantivy::{query::AllQuery, schema::Value, DateTime, Order}; use tantivy::{query::AllQuery, schema::Value, DateTime, Order};
use pesde::{names::PackageName, source::pesde::IndexFile};
use crate::{error::Error, package::PackageResponse, AppState}; use crate::{error::Error, package::PackageResponse, AppState};
use pesde::{
names::PackageName,
source::{git_index::GitBasedSource, pesde::IndexFile},
};
#[derive(Deserialize)] #[derive(Deserialize)]
pub struct Request { pub struct Request {

View file

@ -1,6 +1,6 @@
use actix_web::{body::BoxBody, HttpResponse, ResponseError}; use actix_web::{body::BoxBody, HttpResponse, ResponseError};
use log::error; use log::error;
use pesde::source::pesde::errors::ReadFile; use pesde::source::git_index::errors::{ReadFile, RefreshError};
use serde::Serialize; use serde::Serialize;
use thiserror::Error; use thiserror::Error;
@ -31,7 +31,7 @@ pub enum Error {
Git(#[from] git2::Error), Git(#[from] git2::Error),
#[error("failed to refresh source")] #[error("failed to refresh source")]
Refresh(#[from] Box<pesde::source::pesde::errors::RefreshError>), Refresh(#[from] Box<RefreshError>),
#[error("failed to serialize struct")] #[error("failed to serialize struct")]
Serialize(#[from] toml::ser::Error), Serialize(#[from] toml::ser::Error),

View file

@ -67,6 +67,22 @@ impl AddCommand {
PackageSources::Pesde(PesdePackageSource::new(index)) PackageSources::Pesde(PesdePackageSource::new(index))
} }
#[cfg(feature = "wally-compat")]
PackageNames::Wally(_) => {
let index = manifest
.wally_indices
.get(self.index.as_deref().unwrap_or(DEFAULT_INDEX_NAME))
.cloned();
if let Some(index) = self.index.as_ref().filter(|_| index.is_none()) {
log::error!("wally index {index} not found");
return Ok(());
}
let index = index.unwrap_or(read_config()?.default_index);
PackageSources::Wally(pesde::source::wally::WallyPackageSource::new(index))
}
}; };
source source
.refresh(&project) .refresh(&project)
@ -79,6 +95,14 @@ impl AddCommand {
index: self.index, index: self.index,
target: self.target, target: self.target,
}), }),
#[cfg(feature = "wally-compat")]
PackageNames::Wally(name) => DependencySpecifiers::Wally(
pesde::source::wally::specifier::WallyDependencySpecifier {
name: name.clone(),
version: self.name.1.unwrap_or(VersionReq::STAR),
index: self.index,
},
),
}; };
let Some(version_id) = source let Some(version_id) = source
@ -139,6 +163,24 @@ impl AddCommand {
dependency_key dependency_key
); );
} }
#[cfg(feature = "wally-compat")]
DependencySpecifiers::Wally(spec) => {
manifest[dependency_key][alias]["name"] =
toml_edit::value(spec.name.clone().to_string());
manifest[dependency_key][alias]["version"] =
toml_edit::value(format!("^{}", version_id.version()));
if let Some(index) = spec.index.filter(|i| i != DEFAULT_INDEX_NAME) {
manifest[dependency_key][alias]["index"] = toml_edit::value(index);
}
println!(
"added wally {}@{} to {}",
spec.name,
version_id.version(),
dependency_key
);
}
} }
project project

View file

@ -76,7 +76,7 @@ impl Subcommand {
Subcommand::Run(run) => run.run(project), Subcommand::Run(run) => run.run(project),
Subcommand::Install(install) => install.run(project, multi, reqwest), Subcommand::Install(install) => install.run(project, multi, reqwest),
Subcommand::Publish(publish) => publish.run(project, reqwest), Subcommand::Publish(publish) => publish.run(project, reqwest),
Subcommand::SelfInstall(self_install) => self_install.run(project), Subcommand::SelfInstall(self_install) => self_install.run(),
#[cfg(feature = "patches")] #[cfg(feature = "patches")]
Subcommand::Patch(patch) => patch.run(project, reqwest), Subcommand::Patch(patch) => patch.run(project, reqwest),
#[cfg(feature = "patches")] #[cfg(feature = "patches")]

View file

@ -46,6 +46,10 @@ impl OutdatedCommand {
DependencySpecifiers::Pesde(ref mut spec) => { DependencySpecifiers::Pesde(ref mut spec) => {
spec.version = VersionReq::STAR; spec.version = VersionReq::STAR;
} }
#[cfg(feature = "wally-compat")]
DependencySpecifiers::Wally(ref mut spec) => {
spec.version = VersionReq::STAR;
}
}; };
} }

View file

@ -1,14 +1,20 @@
use std::{
io::{Seek, Write},
path::Component,
};
use anyhow::Context; use anyhow::Context;
use clap::Args; use clap::Args;
use colored::Colorize; use colored::Colorize;
use reqwest::StatusCode;
use tempfile::tempfile;
use pesde::{ use pesde::{
manifest::target::Target, manifest::target::Target,
scripts::ScriptName, scripts::ScriptName,
source::{pesde::PesdePackageSource, traits::PackageSource}, source::{pesde::PesdePackageSource, traits::PackageSource},
Project, DEFAULT_INDEX_NAME, MANIFEST_FILE_NAME, MAX_ARCHIVE_SIZE, Project, DEFAULT_INDEX_NAME, MANIFEST_FILE_NAME,
}; };
use reqwest::StatusCode;
use std::path::Component;
#[derive(Debug, Args)] #[derive(Debug, Args)]
pub struct PublishCommand { pub struct PublishCommand {
@ -270,50 +276,26 @@ impl PublishCommand {
println!(); println!();
} }
let temp_manifest_path = project let mut temp_manifest = tempfile().context("failed to create temp manifest file")?;
.data_dir() temp_manifest
.join(format!("temp_manifest_{}", chrono::Utc::now().timestamp())); .write_all(
toml::to_string(&manifest)
std::fs::write( .context("failed to serialize manifest")?
&temp_manifest_path, .as_bytes(),
toml::to_string(&manifest).context("failed to serialize manifest")?, )
) .context("failed to write temp manifest file")?;
.context("failed to write temp manifest file")?; temp_manifest
.rewind()
let mut temp_manifest = std::fs::File::open(&temp_manifest_path) .context("failed to rewind temp manifest file")?;
.context("failed to open temp manifest file")?;
archive.append_file(MANIFEST_FILE_NAME, &mut temp_manifest)?; archive.append_file(MANIFEST_FILE_NAME, &mut temp_manifest)?;
drop(temp_manifest);
std::fs::remove_file(temp_manifest_path)?;
let archive = archive let archive = archive
.into_inner() .into_inner()
.context("failed to encode archive")? .context("failed to encode archive")?
.finish() .finish()
.context("failed to get archive bytes")?; .context("failed to get archive bytes")?;
if archive.len() > MAX_ARCHIVE_SIZE {
anyhow::bail!(
"archive size exceeds maximum size of {} bytes by {} bytes",
MAX_ARCHIVE_SIZE,
archive.len() - MAX_ARCHIVE_SIZE
);
}
if self.dry_run {
std::fs::write("package.tar.gz", archive)?;
println!(
"{}",
"(dry run) package written to package.tar.gz".green().bold()
);
return Ok(());
}
let source = PesdePackageSource::new( let source = PesdePackageSource::new(
manifest manifest
.indices .indices
@ -328,6 +310,25 @@ impl PublishCommand {
.config(&project) .config(&project)
.context("failed to get source config")?; .context("failed to get source config")?;
if archive.len() > config.max_archive_size {
anyhow::bail!(
"archive size exceeds maximum size of {} bytes by {} bytes",
config.max_archive_size,
archive.len() - config.max_archive_size
);
}
if self.dry_run {
std::fs::write("package.tar.gz", archive)?;
println!(
"{}",
"(dry run) package written to package.tar.gz".green().bold()
);
return Ok(());
}
match reqwest match reqwest
.post(format!("{}/v0/packages", config.api())) .post(format!("{}/v0/packages", config.api()))
.multipart(reqwest::blocking::multipart::Form::new().part( .multipart(reqwest::blocking::multipart::Form::new().part(

View file

@ -1,9 +1,7 @@
use crate::cli::{bin_dir, scripts::update_scripts_folder, version::update_bin_exe, HOME_DIR}; use crate::cli::{bin_dir, version::update_bin_exe, HOME_DIR};
use anyhow::Context; use anyhow::Context;
use clap::Args; use clap::Args;
use colored::Colorize; use colored::Colorize;
use pesde::Project;
#[derive(Debug, Args)] #[derive(Debug, Args)]
pub struct SelfInstallCommand { pub struct SelfInstallCommand {
/// Skip adding the bin directory to the PATH /// Skip adding the bin directory to the PATH
@ -13,9 +11,7 @@ pub struct SelfInstallCommand {
} }
impl SelfInstallCommand { impl SelfInstallCommand {
pub fn run(self, project: Project) -> anyhow::Result<()> { pub fn run(self) -> anyhow::Result<()> {
update_scripts_folder(&project)?;
let bin_dir = bin_dir()?; let bin_dir = bin_dir()?;
#[cfg(windows)] #[cfg(windows)]
@ -30,10 +26,8 @@ impl SelfInstallCommand {
let path: String = env.get_value("Path").context("failed to get Path value")?; let path: String = env.get_value("Path").context("failed to get Path value")?;
let bin_dir = bin_dir.to_string_lossy(); let bin_dir = bin_dir.to_string_lossy();
let exists = path let exists = path.split(';').any(|part| *part == bin_dir);
.split(';')
.any(|part| *part == bin_dir);
if !exists { if !exists {
let new_path = format!("{path};{bin_dir}"); let new_path = format!("{path};{bin_dir}");

View file

@ -1,94 +1,31 @@
use crate::{ use std::fs::remove_dir_all;
cli::{config::read_config, home_dir},
util::authenticate_conn,
};
use anyhow::Context; use anyhow::Context;
use gix::remote::Direction;
use pesde::Project; use pesde::Project;
pub fn update_scripts_folder(project: &Project) -> anyhow::Result<()> { use crate::cli::{config::read_config, home_dir};
pub fn update_scripts_folder(_project: &Project) -> anyhow::Result<()> {
let scripts_dir = home_dir()?.join("scripts"); let scripts_dir = home_dir()?.join("scripts");
if scripts_dir.exists() { if scripts_dir.exists() {
let repo = gix::open(&scripts_dir).context("failed to open scripts repository")?; // checking out the repository seems to be corrupting the repository contents
// TODO: add actual `git pull`-esque functionality
remove_dir_all(&scripts_dir).context("failed to remove scripts directory")?;
}
let remote = repo std::fs::create_dir_all(&scripts_dir).context("failed to create scripts directory")?;
.find_default_remote(Direction::Fetch)
.context("missing default remote of scripts repository")?
.context("failed to find default remote of scripts repository")?;
let mut connection = remote let cli_config = read_config()?;
.connect(Direction::Fetch)
.context("failed to connect to default remote of scripts repository")?;
authenticate_conn(&mut connection, project.auth_config()); gix::prepare_clone(cli_config.scripts_repo, &scripts_dir)
.context("failed to prepare scripts repository clone")?
let results = connection .fetch_then_checkout(gix::progress::Discard, &false.into())
.prepare_fetch(gix::progress::Discard, Default::default()) .context("failed to fetch and checkout scripts repository")?
.context("failed to prepare scripts repository fetch")? .0
.receive(gix::progress::Discard, &false.into()) .main_worktree(gix::progress::Discard, &false.into())
.context("failed to receive new scripts repository contents")?; .context("failed to set scripts repository as main worktree")?;
let remote_ref = results
.ref_map
.remote_refs
.first()
.context("failed to get remote refs of scripts repository")?;
let unpacked = remote_ref.unpack();
let oid = unpacked
.1
.or(unpacked.2)
.context("couldn't find oid in remote ref")?;
let tree = repo
.find_object(oid)
.context("failed to find scripts repository tree")?
.peel_to_tree()
.context("failed to peel scripts repository object to tree")?;
let mut index = gix::index::File::from_state(
gix::index::State::from_tree(&tree.id, &repo.objects, Default::default())
.context("failed to create index state from scripts repository tree")?,
repo.index_path(),
);
let opts = gix::worktree::state::checkout::Options {
overwrite_existing: true,
destination_is_initially_empty: false,
..Default::default()
};
gix::worktree::state::checkout(
&mut index,
repo.work_dir().context("scripts repo is bare")?,
repo.objects
.clone()
.into_arc()
.context("failed to clone objects")?,
&gix::progress::Discard,
&gix::progress::Discard,
&false.into(),
opts,
)
.context("failed to checkout scripts repository")?;
index
.write(gix::index::write::Options::default())
.context("failed to write index")?;
} else {
std::fs::create_dir_all(&scripts_dir).context("failed to create scripts directory")?;
let cli_config = read_config()?;
gix::prepare_clone(cli_config.scripts_repo, &scripts_dir)
.context("failed to prepare scripts repository clone")?
.fetch_then_checkout(gix::progress::Discard, &false.into())
.context("failed to fetch and checkout scripts repository")?
.0
.main_worktree(gix::progress::Discard, &false.into())
.context("failed to set scripts repository as main worktree")?;
};
Ok(()) Ok(())
} }

View file

@ -110,11 +110,11 @@ pub mod errors {
#[derive(Debug, Error)] #[derive(Debug, Error)]
#[non_exhaustive] #[non_exhaustive]
pub enum DownloadGraphError { pub enum DownloadGraphError {
/// Error occurred deserializing the project manifest /// An error occurred deserializing the project manifest
#[error("error deserializing project manifest")] #[error("error deserializing project manifest")]
ManifestDeserializationFailed(#[from] crate::errors::ManifestReadError), ManifestDeserializationFailed(#[from] crate::errors::ManifestReadError),
/// Error occurred refreshing a package source /// An error occurred refreshing a package source
#[error("failed to refresh package source")] #[error("failed to refresh package source")]
RefreshFailed(#[from] Box<crate::source::errors::RefreshError>), RefreshFailed(#[from] Box<crate::source::errors::RefreshError>),

View file

@ -38,13 +38,12 @@ pub const LOCKFILE_FILE_NAME: &str = "pesde.lock";
pub const DEFAULT_INDEX_NAME: &str = "default"; pub const DEFAULT_INDEX_NAME: &str = "default";
/// The name of the packages container /// The name of the packages container
pub const PACKAGES_CONTAINER_NAME: &str = ".pesde"; pub const PACKAGES_CONTAINER_NAME: &str = ".pesde";
/// Maximum size of a package's archive pub(crate) const LINK_LIB_NO_FILE_FOUND: &str = "____pesde_no_export_file_found";
pub const MAX_ARCHIVE_SIZE: usize = 4 * 1024 * 1024;
/// Struct containing the authentication configuration /// Struct containing the authentication configuration
#[derive(Debug, Default, Clone)] #[derive(Debug, Default, Clone)]
pub struct AuthConfig { pub struct AuthConfig {
pesde_token: Option<String>, github_token: Option<String>,
git_credentials: Option<gix::sec::identity::Account>, git_credentials: Option<gix::sec::identity::Account>,
} }
@ -54,9 +53,9 @@ impl AuthConfig {
AuthConfig::default() AuthConfig::default()
} }
/// Access the pesde token /// Access the GitHub token
pub fn pesde_token(&self) -> Option<&str> { pub fn github_token(&self) -> Option<&str> {
self.pesde_token.as_deref() self.github_token.as_deref()
} }
/// Access the git credentials /// Access the git credentials
@ -64,9 +63,9 @@ impl AuthConfig {
self.git_credentials.as_ref() self.git_credentials.as_ref()
} }
/// Set the pesde token /// Set the GitHub token
pub fn with_pesde_token<S: AsRef<str>>(mut self, token: Option<S>) -> Self { pub fn with_github_token<S: AsRef<str>>(mut self, token: Option<S>) -> Self {
self.pesde_token = token.map(|s| s.as_ref().to_string()); self.github_token = token.map(|s| s.as_ref().to_string());
self self
} }

View file

@ -1,3 +1,9 @@
use std::{
collections::BTreeMap,
fs::create_dir_all,
path::{Path, PathBuf},
};
use crate::{ use crate::{
linking::generator::get_file_types, linking::generator::get_file_types,
lockfile::DownloadedGraph, lockfile::DownloadedGraph,
@ -5,12 +11,7 @@ use crate::{
names::PackageNames, names::PackageNames,
scripts::{execute_script, ScriptName}, scripts::{execute_script, ScriptName},
source::{fs::store_in_cas, traits::PackageRef, version_id::VersionId}, source::{fs::store_in_cas, traits::PackageRef, version_id::VersionId},
Project, PACKAGES_CONTAINER_NAME, Project, LINK_LIB_NO_FILE_FOUND, PACKAGES_CONTAINER_NAME,
};
use std::{
collections::BTreeMap,
fs::create_dir_all,
path::{Path, PathBuf},
}; };
/// Generates linking modules for a project /// Generates linking modules for a project
@ -23,7 +24,7 @@ fn create_and_canonicalize<P: AsRef<Path>>(path: P) -> std::io::Result<PathBuf>
} }
fn write_cas(destination: PathBuf, cas_dir: &Path, contents: &str) -> std::io::Result<()> { fn write_cas(destination: PathBuf, cas_dir: &Path, contents: &str) -> std::io::Result<()> {
let cas_path = store_in_cas(cas_dir, contents)?.1; let cas_path = store_in_cas(cas_dir, contents.as_bytes())?.1;
std::fs::hard_link(cas_path, destination) std::fs::hard_link(cas_path, destination)
} }
@ -50,37 +51,45 @@ impl Project {
version_id.version(), version_id.version(),
); );
let lib_file = lib_file.to_path(&container_folder); let types = if lib_file.as_str() != LINK_LIB_NO_FILE_FOUND {
let lib_file = lib_file.to_path(&container_folder);
let contents = match std::fs::read_to_string(&lib_file) { let contents = match std::fs::read_to_string(&lib_file) {
Ok(contents) => contents, Ok(contents) => contents,
Err(e) if e.kind() == std::io::ErrorKind::NotFound => { Err(e) if e.kind() == std::io::ErrorKind::NotFound => {
return Err(errors::LinkingError::LibFileNotFound( return Err(errors::LinkingError::LibFileNotFound(
lib_file.display().to_string(), lib_file.display().to_string(),
)); ));
} }
Err(e) => return Err(e.into()), Err(e) => return Err(e.into()),
};
let types = match get_file_types(&contents) {
Ok(types) => types,
Err(e) => {
return Err(errors::LinkingError::FullMoon(
lib_file.display().to_string(),
e,
))
}
};
log::debug!("{name}@{version_id} has {} exported types", types.len());
types
} else {
vec![]
}; };
let types = match get_file_types(&contents) {
Ok(types) => types,
Err(e) => {
return Err(errors::LinkingError::FullMoon(
lib_file.display().to_string(),
e,
))
}
};
log::debug!("{name}@{version_id} has {} exported types", types.len());
package_types package_types
.entry(name) .entry(name)
.or_default() .or_default()
.insert(version_id, types); .insert(version_id, types);
#[cfg(feature = "roblox")] #[cfg(feature = "roblox")]
if let Target::Roblox { build_files, .. } = &node.target { if let Some(Target::Roblox { build_files, .. }) =
Some(&node.target).filter(|_| !node.node.pkg_ref.is_wally())
{
let script_name = ScriptName::RobloxSyncConfigGenerator.to_string(); let script_name = ScriptName::RobloxSyncConfigGenerator.to_string();
let Some(script_path) = manifest.scripts.get(&script_name) else { let Some(script_path) = manifest.scripts.get(&script_name) else {

View file

@ -1,6 +1,7 @@
use crate::cli::{ use crate::cli::{
auth::get_token, auth::get_token,
home_dir, home_dir,
scripts::update_scripts_folder,
version::{check_for_updates, current_version, get_or_download_version, max_installed_version}, version::{check_for_updates, current_version, get_or_download_version, max_installed_version},
HOME_DIR, HOME_DIR,
}; };
@ -121,7 +122,7 @@ fn run() -> anyhow::Result<()> {
cwd, cwd,
data_dir, data_dir,
cas_dir, cas_dir,
AuthConfig::new().with_pesde_token(token.as_ref()), AuthConfig::new().with_github_token(token.as_ref()),
); );
let reqwest = { let reqwest = {
@ -161,6 +162,15 @@ fn run() -> anyhow::Result<()> {
); );
} }
} }
match update_scripts_folder(&project) {
Ok(_) => {}
Err(e) => {
println!(
"{}",
format!("failed to update scripts: {e}\n\n").red().bold()
);
}
}
let target_version = project let target_version = project
.deser_manifest() .deser_manifest()

View file

@ -81,6 +81,9 @@ impl PackageName {
pub enum PackageNames { pub enum PackageNames {
/// A pesde package name /// A pesde package name
Pesde(PackageName), Pesde(PackageName),
/// A Wally package name
#[cfg(feature = "wally-compat")]
Wally(wally::WallyPackageName),
} }
impl PackageNames { impl PackageNames {
@ -88,6 +91,8 @@ impl PackageNames {
pub fn as_str(&self) -> (&str, &str) { pub fn as_str(&self) -> (&str, &str) {
match self { match self {
PackageNames::Pesde(name) => name.as_str(), PackageNames::Pesde(name) => name.as_str(),
#[cfg(feature = "wally-compat")]
PackageNames::Wally(name) => name.as_str(),
} }
} }
@ -95,6 +100,8 @@ impl PackageNames {
pub fn escaped(&self) -> String { pub fn escaped(&self) -> String {
match self { match self {
PackageNames::Pesde(name) => name.escaped(), PackageNames::Pesde(name) => name.escaped(),
#[cfg(feature = "wally-compat")]
PackageNames::Wally(name) => name.escaped(),
} }
} }
} }
@ -103,6 +110,8 @@ impl Display for PackageNames {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self { match self {
PackageNames::Pesde(name) => write!(f, "{name}"), PackageNames::Pesde(name) => write!(f, "{name}"),
#[cfg(feature = "wally-compat")]
PackageNames::Wally(name) => write!(f, "{name}"),
} }
} }
} }
@ -111,6 +120,15 @@ impl FromStr for PackageNames {
type Err = errors::PackageNamesError; type Err = errors::PackageNamesError;
fn from_str(s: &str) -> Result<Self, Self::Err> { fn from_str(s: &str) -> Result<Self, Self::Err> {
#[cfg(feature = "wally-compat")]
if let Some(wally_name) = s
.strip_prefix("wally#")
.or_else(|| if s.contains('-') { Some(s) } else { None })
.and_then(|s| wally::WallyPackageName::from_str(s).ok())
{
return Ok(PackageNames::Wally(wally_name));
}
if let Ok(name) = PackageName::from_str(s) { if let Ok(name) = PackageName::from_str(s) {
Ok(PackageNames::Pesde(name)) Ok(PackageNames::Pesde(name))
} else { } else {
@ -119,6 +137,64 @@ impl FromStr for PackageNames {
} }
} }
/// Wally package names
#[cfg(feature = "wally-compat")]
pub mod wally {
use std::{fmt::Display, str::FromStr};
use serde_with::{DeserializeFromStr, SerializeDisplay};
use crate::names::{errors, ErrorReason};
/// A Wally package name
#[derive(
Debug, DeserializeFromStr, SerializeDisplay, Clone, PartialEq, Eq, Hash, PartialOrd, Ord,
)]
pub struct WallyPackageName(String, String);
impl FromStr for WallyPackageName {
type Err = errors::WallyPackageNameError;
fn from_str(s: &str) -> Result<Self, Self::Err> {
let (scope, name) = s
.strip_prefix("wally#")
.unwrap_or(s)
.split_once('/')
.ok_or(Self::Err::InvalidFormat(s.to_string()))?;
for (reason, part) in [(ErrorReason::Scope, scope), (ErrorReason::Name, name)] {
if part.is_empty() || part.len() > 64 {
return Err(Self::Err::InvalidLength(reason, part.to_string()));
}
if !part.chars().all(|c| c.is_ascii_alphanumeric() || c == '-') {
return Err(Self::Err::InvalidCharacters(reason, part.to_string()));
}
}
Ok(Self(scope.to_string(), name.to_string()))
}
}
impl Display for WallyPackageName {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "wally#{}/{}", self.0, self.1)
}
}
impl WallyPackageName {
/// Returns the parts of the package name
pub fn as_str(&self) -> (&str, &str) {
(&self.0, &self.1)
}
/// Returns the package name as a string suitable for use in the filesystem
pub fn escaped(&self) -> String {
format!("wally#{}+{}", self.0, self.1)
}
}
}
/// Errors that can occur when working with package names /// Errors that can occur when working with package names
pub mod errors { pub mod errors {
use thiserror::Error; use thiserror::Error;
@ -149,11 +225,28 @@ pub mod errors {
InvalidLength(ErrorReason, String), InvalidLength(ErrorReason, String),
} }
/// Errors that can occur when working with Wally package names
#[cfg(feature = "wally-compat")]
#[derive(Debug, Error)]
pub enum WallyPackageNameError {
/// The package name is not in the format `scope/name`
#[error("wally package name `{0}` is not in the format `scope/name`")]
InvalidFormat(String),
/// The package name is outside the allowed characters: a-z, 0-9, and -
#[error("wally package {0} `{1}` contains characters outside a-z, 0-9, and -")]
InvalidCharacters(ErrorReason, String),
/// The package name is not within 1-64 characters long
#[error("wally package {0} `{1}` is not within 1-64 characters long")]
InvalidLength(ErrorReason, String),
}
/// Errors that can occur when working with package names /// Errors that can occur when working with package names
#[derive(Debug, Error)] #[derive(Debug, Error)]
#[non_exhaustive] #[non_exhaustive]
pub enum PackageNamesError { pub enum PackageNamesError {
/// The pesde package name is invalid /// The package name is invalid
#[error("invalid package name {0}")] #[error("invalid package name {0}")]
InvalidPackageName(String), InvalidPackageName(String),
} }

View file

@ -152,6 +152,30 @@ impl Project {
PackageSources::Pesde(PesdePackageSource::new(index_url)) PackageSources::Pesde(PesdePackageSource::new(index_url))
} }
#[cfg(feature = "wally-compat")]
DependencySpecifiers::Wally(specifier) => {
let index_url = if depth == 0 || overridden {
let index_name = specifier.index.as_deref().unwrap_or(DEFAULT_INDEX_NAME);
manifest
.wally_indices
.get(index_name)
.ok_or(errors::DependencyGraphError::WallyIndexNotFound(
index_name.to_string(),
))?
.clone()
} else {
let index_url = specifier.index.clone().unwrap();
index_url
.clone()
.try_into()
// specifiers in indices store the index url in this field
.unwrap()
};
PackageSources::Wally(crate::source::wally::WallyPackageSource::new(index_url))
}
}; };
if refreshed_sources.insert(source.clone()) { if refreshed_sources.insert(source.clone()) {
@ -305,19 +329,24 @@ pub mod errors {
/// An error occurred while deserializing the manifest /// An error occurred while deserializing the manifest
#[error("failed to deserialize manifest")] #[error("failed to deserialize manifest")]
ManifestRead(#[from] crate::errors::ManifestReadError), ManifestRead(#[from] crate::errors::ManifestReadError),
/// An error occurred while reading all dependencies from the manifest /// An error occurred while reading all dependencies from the manifest
#[error("error getting all project dependencies")] #[error("error getting all project dependencies")]
AllDependencies(#[from] crate::manifest::errors::AllDependenciesError), AllDependencies(#[from] crate::manifest::errors::AllDependenciesError),
/// An index was not found in the manifest /// An index was not found in the manifest
#[error("index named {0} not found in manifest")] #[error("index named `{0}` not found in manifest")]
IndexNotFound(String), IndexNotFound(String),
/// A Wally index was not found in the manifest
#[cfg(feature = "wally-compat")]
#[error("wally index named `{0}` not found in manifest")]
WallyIndexNotFound(String),
/// An error occurred while refreshing a package source /// An error occurred while refreshing a package source
#[error("error refreshing package source")] #[error("error refreshing package source")]
Refresh(#[from] crate::source::errors::RefreshError), Refresh(#[from] crate::source::errors::RefreshError),
/// An error occurred while resolving a package /// An error occurred while resolving a package
#[error("error resolving package")] #[error("error resolving package")]
Resolve(#[from] crate::source::errors::ResolveError), Resolve(#[from] crate::source::errors::ResolveError),

View file

@ -1,13 +1,12 @@
use std::{ use std::{
ffi::OsStr, ffi::OsStr,
fmt::{Display, Formatter},
io::{BufRead, BufReader}, io::{BufRead, BufReader},
path::Path, path::Path,
process::{Command, Stdio}, process::{Command, Stdio},
thread::spawn, thread::spawn,
}; };
use std::fmt::{Display, Formatter};
/// Script names used by pesde /// Script names used by pesde
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)] #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum ScriptName { pub enum ScriptName {
@ -41,6 +40,7 @@ pub fn execute_script<A: IntoIterator<Item = S>, S: AsRef<OsStr>, P: AsRef<Path>
match Command::new("lune") match Command::new("lune")
.arg("run") .arg("run")
.arg(script_path.as_os_str()) .arg(script_path.as_os_str())
.arg("--")
.args(args) .args(args)
.current_dir(cwd) .current_dir(cwd)
.stdin(Stdio::null()) .stdin(Stdio::null())
@ -78,11 +78,11 @@ pub fn execute_script<A: IntoIterator<Item = S>, S: AsRef<OsStr>, P: AsRef<Path>
for line in stdout.lines() { for line in stdout.lines() {
match line { match line {
Ok(line) => { Ok(line) => {
log::info!("[{script_2}]: {line}");
if return_stdout { if return_stdout {
stdout_str.push_str(&line); stdout_str.push_str(&line);
stdout_str.push('\n'); stdout_str.push('\n');
} else {
log::info!("[{script_2}]: {line}");
} }
} }
Err(e) => { Err(e) => {

View file

@ -24,9 +24,9 @@ pub struct PackageFS(pub(crate) BTreeMap<RelativePathBuf, FSEntry>);
pub(crate) fn store_in_cas<P: AsRef<Path>>( pub(crate) fn store_in_cas<P: AsRef<Path>>(
cas_dir: P, cas_dir: P,
contents: &str, contents: &[u8],
) -> std::io::Result<(String, PathBuf)> { ) -> std::io::Result<(String, PathBuf)> {
let hash = hash(contents.as_bytes()); let hash = hash(contents);
let (prefix, rest) = hash.split_at(2); let (prefix, rest) = hash.split_at(2);
let folder = cas_dir.as_ref().join(prefix); let folder = cas_dir.as_ref().join(prefix);
@ -74,4 +74,15 @@ impl PackageFS {
Ok(()) Ok(())
} }
/// Returns the contents of the file with the given hash
pub fn read_file<P: AsRef<Path>, H: AsRef<str>>(
&self,
file_hash: H,
cas_path: P,
) -> Option<String> {
let (prefix, rest) = file_hash.as_ref().split_at(2);
let cas_file_path = cas_path.as_ref().join(prefix).join(rest);
std::fs::read_to_string(cas_file_path).ok()
}
} }

268
src/source/git_index.rs Normal file
View file

@ -0,0 +1,268 @@
use gix::remote::Direction;
use crate::{util::authenticate_conn, Project};
/// A trait for sources that are based on Git repositories
pub trait GitBasedSource {
/// The path to the index
fn path(&self, project: &Project) -> std::path::PathBuf;
/// The URL of the repository
fn repo_url(&self) -> &gix::Url;
/// Gets the tree of the repository
fn tree<'a>(&'a self, repo: &'a gix::Repository) -> Result<gix::Tree, errors::TreeError> {
// this is a bare repo, so this is the actual path
let path = repo.path().to_path_buf();
let remote = match repo.find_default_remote(Direction::Fetch) {
Some(Ok(remote)) => remote,
Some(Err(e)) => return Err(errors::TreeError::GetDefaultRemote(path, Box::new(e))),
None => {
return Err(errors::TreeError::NoDefaultRemote(path));
}
};
let refspec = match remote.refspecs(Direction::Fetch).first() {
Some(head) => head,
None => return Err(errors::TreeError::NoRefSpecs(path)),
};
let spec_ref = refspec.to_ref();
let local_ref = match spec_ref.local() {
Some(local) => local
.to_string()
.replace('*', repo.branch_names().first().unwrap_or(&"main")),
None => return Err(errors::TreeError::NoLocalRefSpec(path)),
};
let reference = match repo.find_reference(&local_ref) {
Ok(reference) => reference,
Err(e) => return Err(errors::TreeError::NoReference(local_ref.to_string(), e)),
};
let reference_name = reference.name().as_bstr().to_string();
let id = match reference.into_fully_peeled_id() {
Ok(id) => id,
Err(e) => return Err(errors::TreeError::CannotPeel(reference_name, e)),
};
let id_str = id.to_string();
let object = match id.object() {
Ok(object) => object,
Err(e) => return Err(errors::TreeError::CannotConvertToObject(id_str, e)),
};
match object.peel_to_tree() {
Ok(tree) => Ok(tree),
Err(e) => Err(errors::TreeError::CannotPeelToTree(id_str, e)),
}
}
/// Reads a file from the repository
fn read_file<I: IntoIterator<Item = P> + Clone, P: ToString + PartialEq<gix::bstr::BStr>>(
&self,
file_path: I,
project: &Project,
) -> Result<Option<String>, errors::ReadFile> {
let path = self.path(project);
let repo = match gix::open(&path) {
Ok(repo) => repo,
Err(e) => return Err(errors::ReadFile::Open(path, Box::new(e))),
};
let tree = match self.tree(&repo) {
Ok(tree) => tree,
Err(e) => return Err(errors::ReadFile::Tree(path, Box::new(e))),
};
let file_path_str = file_path
.clone()
.into_iter()
.map(|s| s.to_string())
.collect::<Vec<_>>()
.join(std::path::MAIN_SEPARATOR_STR);
let mut lookup_buf = vec![];
let entry = match tree.lookup_entry(file_path, &mut lookup_buf) {
Ok(Some(entry)) => entry,
Ok(None) => return Ok(None),
Err(e) => return Err(errors::ReadFile::Lookup(file_path_str, e)),
};
let object = match entry.object() {
Ok(object) => object,
Err(e) => return Err(errors::ReadFile::Lookup(file_path_str, e)),
};
let blob = object.into_blob();
let string = String::from_utf8(blob.data.clone())
.map_err(|e| errors::ReadFile::Utf8(file_path_str, e))?;
Ok(Some(string))
}
/// Refreshes the repository
fn refresh(&self, project: &Project) -> Result<(), errors::RefreshError> {
let path = self.path(project);
if path.exists() {
let repo = match gix::open(&path) {
Ok(repo) => repo,
Err(e) => return Err(errors::RefreshError::Open(path, Box::new(e))),
};
let remote = match repo.find_default_remote(Direction::Fetch) {
Some(Ok(remote)) => remote,
Some(Err(e)) => {
return Err(errors::RefreshError::GetDefaultRemote(path, Box::new(e)))
}
None => {
return Err(errors::RefreshError::NoDefaultRemote(path));
}
};
let mut connection = remote.connect(Direction::Fetch).map_err(|e| {
errors::RefreshError::Connect(self.repo_url().to_string(), Box::new(e))
})?;
authenticate_conn(&mut connection, &project.auth_config);
connection
.prepare_fetch(gix::progress::Discard, Default::default())
.map_err(|e| {
errors::RefreshError::PrepareFetch(self.repo_url().to_string(), Box::new(e))
})?
.receive(gix::progress::Discard, &false.into())
.map_err(|e| {
errors::RefreshError::Read(self.repo_url().to_string(), Box::new(e))
})?;
return Ok(());
}
std::fs::create_dir_all(&path)?;
let auth_config = project.auth_config.clone();
gix::prepare_clone_bare(self.repo_url().clone(), &path)
.map_err(|e| errors::RefreshError::Clone(self.repo_url().to_string(), Box::new(e)))?
.configure_connection(move |c| {
authenticate_conn(c, &auth_config);
Ok(())
})
.fetch_only(gix::progress::Discard, &false.into())
.map_err(|e| errors::RefreshError::Fetch(self.repo_url().to_string(), Box::new(e)))?;
Ok(())
}
}
/// Errors that can occur when interacting with a git-based package source
pub mod errors {
use std::path::PathBuf;
use thiserror::Error;
/// Errors that can occur when refreshing a git-based package source
#[derive(Debug, Error)]
#[non_exhaustive]
pub enum RefreshError {
/// Error interacting with the filesystem
#[error("error interacting with the filesystem")]
Io(#[from] std::io::Error),
/// Error opening the repository
#[error("error opening repository at {0}")]
Open(PathBuf, #[source] Box<gix::open::Error>),
/// No default remote found in repository
#[error("no default remote found in repository at {0}")]
NoDefaultRemote(PathBuf),
/// Error getting default remote from repository
#[error("error getting default remote from repository at {0}")]
GetDefaultRemote(PathBuf, #[source] Box<gix::remote::find::existing::Error>),
/// Error connecting to remote repository
#[error("error connecting to remote repository at {0}")]
Connect(String, #[source] Box<gix::remote::connect::Error>),
/// Error preparing fetch from remote repository
#[error("error preparing fetch from remote repository at {0}")]
PrepareFetch(String, #[source] Box<gix::remote::fetch::prepare::Error>),
/// Error reading from remote repository
#[error("error reading from remote repository at {0}")]
Read(String, #[source] Box<gix::remote::fetch::Error>),
/// Error cloning repository
#[error("error cloning repository from {0}")]
Clone(String, #[source] Box<gix::clone::Error>),
/// Error fetching repository
#[error("error fetching repository from {0}")]
Fetch(String, #[source] Box<gix::clone::fetch::Error>),
}
/// Errors that can occur when reading a git-based package source's tree
#[derive(Debug, Error)]
#[non_exhaustive]
pub enum TreeError {
/// Error interacting with the filesystem
#[error("error interacting with the filesystem")]
Io(#[from] std::io::Error),
/// No default remote found in repository
#[error("no default remote found in repository at {0}")]
NoDefaultRemote(PathBuf),
/// Error getting default remote from repository
#[error("error getting default remote from repository at {0}")]
GetDefaultRemote(PathBuf, #[source] Box<gix::remote::find::existing::Error>),
/// Error getting refspec from remote repository
#[error("no refspecs found in repository at {0}")]
NoRefSpecs(PathBuf),
/// Error getting local refspec from remote repository
#[error("no local refspec found in repository at {0}")]
NoLocalRefSpec(PathBuf),
/// Error finding reference in repository
#[error("no reference found for local refspec {0}")]
NoReference(String, #[source] gix::reference::find::existing::Error),
/// Error peeling reference in repository
#[error("cannot peel reference {0}")]
CannotPeel(String, #[source] gix::reference::peel::Error),
/// Error converting id to object in repository
#[error("error converting id {0} to object")]
CannotConvertToObject(String, #[source] gix::object::find::existing::Error),
/// Error peeling object to tree in repository
#[error("error peeling object {0} to tree")]
CannotPeelToTree(String, #[source] gix::object::peel::to_kind::Error),
}
/// Errors that can occur when reading a file from a git-based package source
#[derive(Debug, Error)]
#[non_exhaustive]
pub enum ReadFile {
/// Error opening the repository
#[error("error opening repository at {0}")]
Open(PathBuf, #[source] Box<gix::open::Error>),
/// Error reading tree from repository
#[error("error getting tree from repository at {0}")]
Tree(PathBuf, #[source] Box<TreeError>),
/// Error looking up entry in tree
#[error("error looking up entry {0} in tree")]
Lookup(String, #[source] gix::object::find::existing::Error),
/// Error reading file as utf8
#[error("error parsing file for {0} as utf8")]
Utf8(String, #[source] std::string::FromUtf8Error),
}
}

View file

@ -12,6 +12,8 @@ use crate::{
/// Packages' filesystems /// Packages' filesystems
pub mod fs; pub mod fs;
/// Git index-based package source utilities
pub mod git_index;
/// The pesde package source /// The pesde package source
pub mod pesde; pub mod pesde;
/// Package references /// Package references
@ -22,6 +24,9 @@ pub mod specifiers;
pub mod traits; pub mod traits;
/// Version IDs /// Version IDs
pub mod version_id; pub mod version_id;
/// The Wally package source
#[cfg(feature = "wally-compat")]
pub mod wally;
/// The result of resolving a package /// The result of resolving a package
pub type ResolveResult<Ref> = (PackageNames, BTreeMap<VersionId, Ref>); pub type ResolveResult<Ref> = (PackageNames, BTreeMap<VersionId, Ref>);
@ -31,6 +36,9 @@ pub type ResolveResult<Ref> = (PackageNames, BTreeMap<VersionId, Ref>);
pub enum PackageSources { pub enum PackageSources {
/// A pesde package source /// A pesde package source
Pesde(pesde::PesdePackageSource), Pesde(pesde::PesdePackageSource),
/// A Wally package source
#[cfg(feature = "wally-compat")]
Wally(wally::WallyPackageSource),
} }
impl PackageSource for PackageSources { impl PackageSource for PackageSources {
@ -43,6 +51,8 @@ impl PackageSource for PackageSources {
fn refresh(&self, project: &Project) -> Result<(), Self::RefreshError> { fn refresh(&self, project: &Project) -> Result<(), Self::RefreshError> {
match self { match self {
PackageSources::Pesde(source) => source.refresh(project).map_err(Into::into), PackageSources::Pesde(source) => source.refresh(project).map_err(Into::into),
#[cfg(feature = "wally-compat")]
PackageSources::Wally(source) => source.refresh(project).map_err(Into::into),
} }
} }
@ -66,6 +76,20 @@ impl PackageSource for PackageSources {
}) })
.map_err(Into::into), .map_err(Into::into),
#[cfg(feature = "wally-compat")]
(PackageSources::Wally(source), DependencySpecifiers::Wally(specifier)) => source
.resolve(specifier, project, project_target)
.map(|(name, results)| {
(
name,
results
.into_iter()
.map(|(version, pkg_ref)| (version, PackageRefs::Wally(pkg_ref)))
.collect(),
)
})
.map_err(Into::into),
_ => Err(errors::ResolveError::Mismatch), _ => Err(errors::ResolveError::Mismatch),
} }
} }
@ -81,6 +105,11 @@ impl PackageSource for PackageSources {
.download(pkg_ref, project, reqwest) .download(pkg_ref, project, reqwest)
.map_err(Into::into), .map_err(Into::into),
#[cfg(feature = "wally-compat")]
(PackageSources::Wally(source), PackageRefs::Wally(pkg_ref)) => source
.download(pkg_ref, project, reqwest)
.map_err(Into::into),
_ => Err(errors::DownloadError::Mismatch), _ => Err(errors::DownloadError::Mismatch),
} }
} }
@ -94,9 +123,9 @@ pub mod errors {
#[derive(Debug, Error)] #[derive(Debug, Error)]
#[non_exhaustive] #[non_exhaustive]
pub enum RefreshError { pub enum RefreshError {
/// The pesde package source failed to refresh /// A git-based package source failed to refresh
#[error("error refreshing pesde package source")] #[error("error refreshing pesde package source")]
Pesde(#[from] crate::source::pesde::errors::RefreshError), GitBased(#[from] crate::source::git_index::errors::RefreshError),
} }
/// Errors that can occur when resolving a package /// Errors that can occur when resolving a package
@ -107,9 +136,14 @@ pub mod errors {
#[error("mismatched dependency specifier for source")] #[error("mismatched dependency specifier for source")]
Mismatch, Mismatch,
/// The pesde package source failed to resolve /// A pesde package source failed to resolve
#[error("error resolving pesde package")] #[error("error resolving pesde package")]
Pesde(#[from] crate::source::pesde::errors::ResolveError), Pesde(#[from] crate::source::pesde::errors::ResolveError),
/// A Wally package source failed to resolve
#[cfg(feature = "wally-compat")]
#[error("error resolving wally package")]
Wally(#[from] crate::source::wally::errors::ResolveError),
} }
/// Errors that can occur when downloading a package /// Errors that can occur when downloading a package
@ -120,8 +154,13 @@ pub mod errors {
#[error("mismatched package ref for source")] #[error("mismatched package ref for source")]
Mismatch, Mismatch,
/// The pesde package source failed to download /// A pesde package source failed to download
#[error("error downloading pesde package")] #[error("error downloading pesde package")]
Pesde(#[from] crate::source::pesde::errors::DownloadError), Pesde(#[from] crate::source::pesde::errors::DownloadError),
/// A Wally package source failed to download
#[cfg(feature = "wally-compat")]
#[error("error downloading wally package")]
Wally(#[from] crate::source::wally::errors::DownloadError),
} }
} }

View file

@ -1,4 +1,18 @@
use gix::remote::Direction; use crate::{
manifest::{
target::{Target, TargetKind},
DependencyType,
},
names::{PackageName, PackageNames},
source::{
fs::{store_in_cas, FSEntry, PackageFS},
git_index::GitBasedSource,
DependencySpecifiers, PackageSource, ResolveResult, VersionId,
},
util::hash,
Project,
};
use gix::Url;
use pkg_ref::PesdePackageRef; use pkg_ref::PesdePackageRef;
use relative_path::RelativePathBuf; use relative_path::RelativePathBuf;
use reqwest::header::ACCEPT; use reqwest::header::ACCEPT;
@ -9,20 +23,7 @@ use std::{
fmt::Debug, fmt::Debug,
hash::Hash, hash::Hash,
io::Read, io::Read,
}; path::PathBuf,
use crate::{
manifest::{
target::{Target, TargetKind},
DependencyType,
},
names::{PackageName, PackageNames},
source::{
fs::{store_in_cas, FSEntry, PackageFS},
DependencySpecifiers, PackageSource, ResolveResult, VersionId,
},
util::{authenticate_conn, hash},
Project,
}; };
/// The pesde package reference /// The pesde package reference
@ -33,7 +34,7 @@ pub mod specifier;
/// The pesde package source /// The pesde package source
#[derive(Debug, Hash, PartialEq, Eq, Clone)] #[derive(Debug, Hash, PartialEq, Eq, Clone)]
pub struct PesdePackageSource { pub struct PesdePackageSource {
repo_url: gix::Url, repo_url: Url,
} }
/// The file containing scope information /// The file containing scope information
@ -46,9 +47,19 @@ pub struct ScopeInfo {
pub owners: BTreeSet<u64>, pub owners: BTreeSet<u64>,
} }
impl GitBasedSource for PesdePackageSource {
fn path(&self, project: &Project) -> PathBuf {
project.data_dir.join("indices").join(hash(self.as_bytes()))
}
fn repo_url(&self) -> &Url {
&self.repo_url
}
}
impl PesdePackageSource { impl PesdePackageSource {
/// Creates a new pesde package source /// Creates a new pesde package source
pub fn new(repo_url: gix::Url) -> Self { pub fn new(repo_url: Url) -> Self {
Self { repo_url } Self { repo_url }
} }
@ -56,114 +67,6 @@ impl PesdePackageSource {
self.repo_url.to_bstring().to_vec() self.repo_url.to_bstring().to_vec()
} }
/// The path to the index
pub fn path(&self, project: &Project) -> std::path::PathBuf {
project.data_dir.join("indices").join(hash(self.as_bytes()))
}
/// The URL of the repository
pub fn repo_url(&self) -> &gix::Url {
&self.repo_url
}
pub(crate) fn tree<'a>(
&'a self,
repo: &'a gix::Repository,
) -> Result<gix::Tree, errors::TreeError> {
// this is a bare repo, so this is the actual path
let path = repo.path().to_path_buf();
let remote = match repo.find_default_remote(Direction::Fetch) {
Some(Ok(remote)) => remote,
Some(Err(e)) => return Err(errors::TreeError::GetDefaultRemote(path, Box::new(e))),
None => {
return Err(errors::TreeError::NoDefaultRemote(path));
}
};
let refspec = match remote.refspecs(Direction::Fetch).first() {
Some(head) => head,
None => return Err(errors::TreeError::NoRefSpecs(path)),
};
let spec_ref = refspec.to_ref();
let local_ref = match spec_ref.local() {
Some(local) => local
.to_string()
.replace('*', repo.branch_names().first().unwrap_or(&"main")),
None => return Err(errors::TreeError::NoLocalRefSpec(path)),
};
let reference = match repo.find_reference(&local_ref) {
Ok(reference) => reference,
Err(e) => return Err(errors::TreeError::NoReference(local_ref.to_string(), e)),
};
let reference_name = reference.name().as_bstr().to_string();
let id = match reference.into_fully_peeled_id() {
Ok(id) => id,
Err(e) => return Err(errors::TreeError::CannotPeel(reference_name, e)),
};
let id_str = id.to_string();
let object = match id.object() {
Ok(object) => object,
Err(e) => return Err(errors::TreeError::CannotConvertToObject(id_str, e)),
};
match object.peel_to_tree() {
Ok(tree) => Ok(tree),
Err(e) => Err(errors::TreeError::CannotPeelToTree(id_str, e)),
}
}
/// Reads a file from the index
pub fn read_file<
I: IntoIterator<Item = P> + Clone,
P: ToString + PartialEq<gix::bstr::BStr>,
>(
&self,
file_path: I,
project: &Project,
) -> Result<Option<String>, errors::ReadFile> {
let path = self.path(project);
let repo = match gix::open(&path) {
Ok(repo) => repo,
Err(e) => return Err(errors::ReadFile::Open(path, Box::new(e))),
};
let tree = match self.tree(&repo) {
Ok(tree) => tree,
Err(e) => return Err(errors::ReadFile::Tree(path, Box::new(e))),
};
let file_path_str = file_path
.clone()
.into_iter()
.map(|s| s.to_string())
.collect::<Vec<_>>()
.join(std::path::MAIN_SEPARATOR_STR);
let mut lookup_buf = vec![];
let entry = match tree.lookup_entry(file_path, &mut lookup_buf) {
Ok(Some(entry)) => entry,
Ok(None) => return Ok(None),
Err(e) => return Err(errors::ReadFile::Lookup(file_path_str, e)),
};
let object = match entry.object() {
Ok(object) => object,
Err(e) => return Err(errors::ReadFile::Lookup(file_path_str, e)),
};
let blob = object.into_blob();
let string = String::from_utf8(blob.data.clone())
.map_err(|e| errors::ReadFile::Utf8(file_path_str, e))?;
Ok(Some(string))
}
/// Reads the config file /// Reads the config file
pub fn config(&self, project: &Project) -> Result<IndexConfig, errors::ConfigError> { pub fn config(&self, project: &Project) -> Result<IndexConfig, errors::ConfigError> {
let file = self.read_file(["config.toml"], project).map_err(Box::new)?; let file = self.read_file(["config.toml"], project).map_err(Box::new)?;
@ -177,9 +80,7 @@ impl PesdePackageSource {
} }
}; };
let config: IndexConfig = toml::from_str(&string)?; toml::from_str(&string).map_err(Into::into)
Ok(config)
} }
/// Reads all packages from the index /// Reads all packages from the index
@ -277,56 +178,12 @@ impl PesdePackageSource {
impl PackageSource for PesdePackageSource { impl PackageSource for PesdePackageSource {
type Specifier = PesdeDependencySpecifier; type Specifier = PesdeDependencySpecifier;
type Ref = PesdePackageRef; type Ref = PesdePackageRef;
type RefreshError = errors::RefreshError; type RefreshError = crate::source::git_index::errors::RefreshError;
type ResolveError = errors::ResolveError; type ResolveError = errors::ResolveError;
type DownloadError = errors::DownloadError; type DownloadError = errors::DownloadError;
fn refresh(&self, project: &Project) -> Result<(), Self::RefreshError> { fn refresh(&self, project: &Project) -> Result<(), Self::RefreshError> {
log::debug!("refreshing pesde index at {}", self.repo_url); GitBasedSource::refresh(self, project)
let path = self.path(project);
if path.exists() {
let repo = match gix::open(&path) {
Ok(repo) => repo,
Err(e) => return Err(Self::RefreshError::Open(path, e)),
};
let remote = match repo.find_default_remote(Direction::Fetch) {
Some(Ok(remote)) => remote,
Some(Err(e)) => return Err(Self::RefreshError::GetDefaultRemote(path, e)),
None => {
return Err(Self::RefreshError::NoDefaultRemote(path));
}
};
let mut connection = remote
.connect(Direction::Fetch)
.map_err(|e| Self::RefreshError::Connect(self.repo_url.clone(), e))?;
authenticate_conn(&mut connection, &project.auth_config);
connection
.prepare_fetch(gix::progress::Discard, Default::default())
.map_err(|e| Self::RefreshError::PrepareFetch(self.repo_url.clone(), e))?
.receive(gix::progress::Discard, &false.into())
.map_err(|e| Self::RefreshError::Read(self.repo_url.clone(), e))?;
return Ok(());
}
std::fs::create_dir_all(&path)?;
let auth_config = project.auth_config.clone();
gix::prepare_clone_bare(self.repo_url.clone(), &path)
.map_err(|e| Self::RefreshError::Clone(self.repo_url.clone(), e))?
.configure_connection(move |c| {
authenticate_conn(c, &auth_config);
Ok(())
})
.fetch_only(gix::progress::Discard, &false.into())
.map_err(|e| Self::RefreshError::Fetch(self.repo_url.clone(), e))?;
Ok(())
} }
fn resolve( fn resolve(
@ -416,7 +273,7 @@ impl PackageSource for PesdePackageSource {
let mut response = reqwest.get(url).header(ACCEPT, "application/octet-stream"); let mut response = reqwest.get(url).header(ACCEPT, "application/octet-stream");
if let Some(token) = &project.auth_config.pesde_token { if let Some(token) = &project.auth_config.github_token {
log::debug!("using token for pesde package download"); log::debug!("using token for pesde package download");
response = response.header("Authorization", format!("Bearer {token}")); response = response.header("Authorization", format!("Bearer {token}"));
} }
@ -439,8 +296,8 @@ impl PackageSource for PesdePackageSource {
continue; continue;
} }
let mut contents = String::new(); let mut contents = vec![];
entry.read_to_string(&mut contents)?; entry.read_to_end(&mut contents)?;
let hash = store_in_cas(&project.cas_dir, &contents)?.0; let hash = store_in_cas(&project.cas_dir, &contents)?.0;
entries.insert(path, FSEntry::File(hash)); entries.insert(path, FSEntry::File(hash));
@ -459,22 +316,32 @@ impl PackageSource for PesdePackageSource {
} }
} }
fn default_archive_size() -> usize {
4 * 1024 * 1024
}
/// The configuration for the pesde index /// The configuration for the pesde index
#[derive(Serialize, Deserialize, Debug, Clone)] #[derive(Deserialize, Debug, Clone)]
#[serde(deny_unknown_fields)] #[serde(deny_unknown_fields)]
pub struct IndexConfig { pub struct IndexConfig {
/// The URL of the API /// The URL of the API
pub api: url::Url, pub api: url::Url,
/// The URL to download packages from /// The URL to download packages from
pub download: Option<String>, pub download: Option<String>,
/// Whether git is allowed as a source for publishing packages /// Whether Git is allowed as a source for publishing packages
#[serde(default)] #[serde(default)]
pub git_allowed: bool, pub git_allowed: bool,
/// Whether other registries are allowed as a source for publishing packages /// Whether other registries are allowed as a source for publishing packages
#[serde(default)] #[serde(default)]
pub other_registries_allowed: bool, pub other_registries_allowed: bool,
/// Whether Wally is allowed as a source for publishing packages
#[serde(default)]
pub wally_allowed: bool,
/// The OAuth client ID for GitHub /// The OAuth client ID for GitHub
pub github_oauth_client_id: String, pub github_oauth_client_id: String,
/// The maximum size of an archive in bytes
#[serde(default = "default_archive_size")]
pub max_archive_size: usize,
} }
impl IndexConfig { impl IndexConfig {
@ -520,112 +387,10 @@ pub type IndexFile = BTreeMap<VersionId, IndexFileEntry>;
pub mod errors { pub mod errors {
use std::path::PathBuf; use std::path::PathBuf;
use crate::source::git_index::errors::{ReadFile, TreeError};
use thiserror::Error; use thiserror::Error;
/// Errors that can occur when refreshing the pesde package source /// Errors that can occur when resolving a package from a pesde package source
#[derive(Debug, Error)]
#[non_exhaustive]
pub enum RefreshError {
/// Error interacting with the filesystem
#[error("error interacting with the filesystem")]
Io(#[from] std::io::Error),
/// Error opening the repository
#[error("error opening repository at {0}")]
Open(PathBuf, #[source] gix::open::Error),
/// No default remote found in repository
#[error("no default remote found in repository at {0}")]
NoDefaultRemote(PathBuf),
/// Error getting default remote from repository
#[error("error getting default remote from repository at {0}")]
GetDefaultRemote(PathBuf, #[source] gix::remote::find::existing::Error),
/// Error connecting to remote repository
#[error("error connecting to remote repository at {0}")]
Connect(gix::Url, #[source] gix::remote::connect::Error),
/// Error preparing fetch from remote repository
#[error("error preparing fetch from remote repository at {0}")]
PrepareFetch(gix::Url, #[source] gix::remote::fetch::prepare::Error),
/// Error reading from remote repository
#[error("error reading from remote repository at {0}")]
Read(gix::Url, #[source] gix::remote::fetch::Error),
/// Error cloning repository
#[error("error cloning repository from {0}")]
Clone(gix::Url, #[source] gix::clone::Error),
/// Error fetching repository
#[error("error fetching repository from {0}")]
Fetch(gix::Url, #[source] gix::clone::fetch::Error),
}
/// Errors that can occur when reading the pesde package source's tree
#[derive(Debug, Error)]
#[non_exhaustive]
pub enum TreeError {
/// Error interacting with the filesystem
#[error("error interacting with the filesystem")]
Io(#[from] std::io::Error),
/// No default remote found in repository
#[error("no default remote found in repository at {0}")]
NoDefaultRemote(PathBuf),
/// Error getting default remote from repository
#[error("error getting default remote from repository at {0}")]
GetDefaultRemote(PathBuf, #[source] Box<gix::remote::find::existing::Error>),
/// Error getting refspec from remote repository
#[error("no refspecs found in repository at {0}")]
NoRefSpecs(PathBuf),
/// Error getting local refspec from remote repository
#[error("no local refspec found in repository at {0}")]
NoLocalRefSpec(PathBuf),
/// Error finding reference in repository
#[error("no reference found for local refspec {0}")]
NoReference(String, #[source] gix::reference::find::existing::Error),
/// Error peeling reference in repository
#[error("cannot peel reference {0}")]
CannotPeel(String, #[source] gix::reference::peel::Error),
/// Error converting id to object in repository
#[error("error converting id {0} to object")]
CannotConvertToObject(String, #[source] gix::object::find::existing::Error),
/// Error peeling object to tree in repository
#[error("error peeling object {0} to tree")]
CannotPeelToTree(String, #[source] gix::object::peel::to_kind::Error),
}
/// Errors that can occur when reading a file from the pesde package source
#[derive(Debug, Error)]
#[non_exhaustive]
pub enum ReadFile {
/// Error opening the repository
#[error("error opening repository at {0}")]
Open(PathBuf, #[source] Box<gix::open::Error>),
/// Error reading tree from repository
#[error("error getting tree from repository at {0}")]
Tree(PathBuf, #[source] Box<TreeError>),
/// Error looking up entry in tree
#[error("error looking up entry {0} in tree")]
Lookup(String, #[source] gix::object::find::existing::Error),
/// Error reading file as utf8
#[error("error parsing file for {0} as utf8")]
Utf8(String, #[source] std::string::FromUtf8Error),
}
/// Errors that can occur when resolving a package from the pesde package source
#[derive(Debug, Error)] #[derive(Debug, Error)]
#[non_exhaustive] #[non_exhaustive]
pub enum ResolveError { pub enum ResolveError {
@ -650,7 +415,7 @@ pub mod errors {
Utf8(String, #[source] std::string::FromUtf8Error), Utf8(String, #[source] std::string::FromUtf8Error),
} }
/// Errors that can occur when reading the config file for the pesde package source /// Errors that can occur when reading the config file for a pesde package source
#[derive(Debug, Error)] #[derive(Debug, Error)]
#[non_exhaustive] #[non_exhaustive]
pub enum ConfigError { pub enum ConfigError {
@ -667,7 +432,7 @@ pub mod errors {
Missing(Box<gix::Url>), Missing(Box<gix::Url>),
} }
/// Errors that can occur when reading all packages from the pesde package source /// Errors that can occur when reading all packages from a pesde package source
#[derive(Debug, Error)] #[derive(Debug, Error)]
#[non_exhaustive] #[non_exhaustive]
pub enum AllPackagesError { pub enum AllPackagesError {
@ -696,7 +461,7 @@ pub mod errors {
Utf8(String, #[source] std::string::FromUtf8Error), Utf8(String, #[source] std::string::FromUtf8Error),
} }
/// Errors that can occur when downloading a package from the pesde package source /// Errors that can occur when downloading a package from a pesde package source
#[derive(Debug, Error)] #[derive(Debug, Error)]
#[non_exhaustive] #[non_exhaustive]
pub enum DownloadError { pub enum DownloadError {

View file

@ -11,30 +11,52 @@ use std::collections::BTreeMap;
pub enum PackageRefs { pub enum PackageRefs {
/// A pesde package reference /// A pesde package reference
Pesde(pesde::pkg_ref::PesdePackageRef), Pesde(pesde::pkg_ref::PesdePackageRef),
/// A Wally package reference
#[cfg(feature = "wally-compat")]
Wally(crate::source::wally::pkg_ref::WallyPackageRef),
}
impl PackageRefs {
/// Returns whether this package reference is a Wally package reference
pub fn is_wally(&self) -> bool {
match self {
#[cfg(feature = "wally-compat")]
PackageRefs::Wally(_) => true,
_ => false,
}
}
} }
impl PackageRef for PackageRefs { impl PackageRef for PackageRefs {
fn dependencies(&self) -> &BTreeMap<String, (DependencySpecifiers, DependencyType)> { fn dependencies(&self) -> &BTreeMap<String, (DependencySpecifiers, DependencyType)> {
match self { match self {
PackageRefs::Pesde(pkg_ref) => pkg_ref.dependencies(), PackageRefs::Pesde(pkg_ref) => pkg_ref.dependencies(),
#[cfg(feature = "wally-compat")]
PackageRefs::Wally(pkg_ref) => pkg_ref.dependencies(),
} }
} }
fn use_new_structure(&self) -> bool { fn use_new_structure(&self) -> bool {
match self { match self {
PackageRefs::Pesde(pkg_ref) => pkg_ref.use_new_structure(), PackageRefs::Pesde(pkg_ref) => pkg_ref.use_new_structure(),
#[cfg(feature = "wally-compat")]
PackageRefs::Wally(pkg_ref) => pkg_ref.use_new_structure(),
} }
} }
fn target_kind(&self) -> TargetKind { fn target_kind(&self) -> TargetKind {
match self { match self {
PackageRefs::Pesde(pkg_ref) => pkg_ref.target_kind(), PackageRefs::Pesde(pkg_ref) => pkg_ref.target_kind(),
#[cfg(feature = "wally-compat")]
PackageRefs::Wally(pkg_ref) => pkg_ref.target_kind(),
} }
} }
fn source(&self) -> PackageSources { fn source(&self) -> PackageSources {
match self { match self {
PackageRefs::Pesde(pkg_ref) => pkg_ref.source(), PackageRefs::Pesde(pkg_ref) => pkg_ref.source(),
#[cfg(feature = "wally-compat")]
PackageRefs::Wally(pkg_ref) => pkg_ref.source(),
} }
} }
} }

View file

@ -8,6 +8,9 @@ use std::fmt::Display;
pub enum DependencySpecifiers { pub enum DependencySpecifiers {
/// A pesde dependency specifier /// A pesde dependency specifier
Pesde(pesde::specifier::PesdeDependencySpecifier), Pesde(pesde::specifier::PesdeDependencySpecifier),
/// A Wally dependency specifier
#[cfg(feature = "wally-compat")]
Wally(crate::source::wally::specifier::WallyDependencySpecifier),
} }
impl DependencySpecifier for DependencySpecifiers {} impl DependencySpecifier for DependencySpecifiers {}
@ -15,6 +18,8 @@ impl Display for DependencySpecifiers {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self { match self {
DependencySpecifiers::Pesde(specifier) => write!(f, "{specifier}"), DependencySpecifiers::Pesde(specifier) => write!(f, "{specifier}"),
#[cfg(feature = "wally-compat")]
DependencySpecifiers::Wally(specifier) => write!(f, "{specifier}"),
} }
} }
} }

View file

@ -0,0 +1,83 @@
use std::path::Path;
use relative_path::RelativePathBuf;
use serde::Deserialize;
use tempfile::TempDir;
use crate::{
manifest::target::Target,
scripts::{execute_script, ScriptName},
Project, LINK_LIB_NO_FILE_FOUND,
};
#[derive(Deserialize)]
#[serde(rename_all = "camelCase")]
struct SourcemapNode {
#[serde(default)]
file_paths: Vec<RelativePathBuf>,
}
pub(crate) fn find_lib_path(
project: &Project,
cwd: &Path,
) -> Result<Option<RelativePathBuf>, errors::FindLibPathError> {
let manifest = project.deser_manifest()?;
let Some(script_path) = manifest
.scripts
.get(&ScriptName::SourcemapGenerator.to_string())
else {
log::warn!("no sourcemap generator script found in manifest");
return Ok(None);
};
let result = execute_script(
Some(&ScriptName::SourcemapGenerator.to_string()),
&script_path.to_path(&project.path),
["--wally"],
cwd,
true,
)?;
if let Some(result) = result {
let node: SourcemapNode = serde_json::from_str(&result)?;
Ok(node.file_paths.into_iter().find(|path| {
path.extension()
.is_some_and(|ext| ext == "lua" || ext == "luau")
}))
} else {
Ok(None)
}
}
pub(crate) fn get_target(
project: &Project,
tempdir: &TempDir,
) -> Result<Target, errors::FindLibPathError> {
Ok(Target::Roblox {
lib: find_lib_path(project, tempdir.path())?
.or_else(|| Some(RelativePathBuf::from(LINK_LIB_NO_FILE_FOUND))),
build_files: Default::default(),
})
}
pub mod errors {
use thiserror::Error;
/// Errors that can occur when finding the lib path
#[derive(Debug, Error)]
#[non_exhaustive]
pub enum FindLibPathError {
/// An error occurred deserializing the project manifest
#[error("error deserializing manifest")]
Manifest(#[from] crate::errors::ManifestReadError),
/// An error occurred while executing the sourcemap generator script
#[error("error executing sourcemap generator script")]
Script(#[from] std::io::Error),
/// An error occurred while deserializing the sourcemap result
#[error("error deserializing sourcemap result")]
Serde(#[from] serde_json::Error),
}
}

View file

@ -0,0 +1,82 @@
use std::collections::BTreeMap;
use semver::{Version, VersionReq};
use serde::{Deserialize, Deserializer};
use crate::{
manifest::{errors, DependencyType},
source::{specifiers::DependencySpecifiers, wally::specifier::WallyDependencySpecifier},
};
#[derive(Deserialize, Clone, Debug)]
#[serde(rename_all = "kebab-case")]
pub struct WallyPackage {
pub version: Version,
}
pub fn deserialize_specifiers<'de, D: Deserializer<'de>>(
deserializer: D,
) -> Result<BTreeMap<String, WallyDependencySpecifier>, D::Error> {
// specifier is in form of `name@version_req`
BTreeMap::<String, String>::deserialize(deserializer)?
.into_iter()
.map(|(k, v)| {
let (name, version) = v.split_once('@').ok_or_else(|| {
serde::de::Error::custom("invalid specifier format, expected `name@version_req`")
})?;
Ok((
k,
WallyDependencySpecifier {
name: name.parse().map_err(serde::de::Error::custom)?,
version: VersionReq::parse(version).map_err(serde::de::Error::custom)?,
index: None,
},
))
})
.collect()
}
#[derive(Deserialize, Clone, Debug)]
#[serde(rename_all = "kebab-case")]
pub struct WallyManifest {
pub package: WallyPackage,
#[serde(default, deserialize_with = "deserialize_specifiers")]
pub dependencies: BTreeMap<String, WallyDependencySpecifier>,
#[serde(default, deserialize_with = "deserialize_specifiers")]
pub server_dependencies: BTreeMap<String, WallyDependencySpecifier>,
#[serde(default, deserialize_with = "deserialize_specifiers")]
pub dev_dependencies: BTreeMap<String, WallyDependencySpecifier>,
}
impl WallyManifest {
/// Get all dependencies from the manifest
pub fn all_dependencies(
&self,
) -> Result<
BTreeMap<String, (DependencySpecifiers, DependencyType)>,
errors::AllDependenciesError,
> {
let mut all_deps = BTreeMap::new();
for (deps, ty) in [
(&self.dependencies, DependencyType::Standard),
(&self.server_dependencies, DependencyType::Standard),
(&self.dev_dependencies, DependencyType::Dev),
] {
for (alias, spec) in deps {
if all_deps
.insert(
alias.clone(),
(DependencySpecifiers::Wally(spec.clone()), ty),
)
.is_some()
{
return Err(errors::AllDependenciesError::AliasConflict(alias.clone()));
}
}
}
Ok(all_deps)
}
}

341
src/source/wally/mod.rs Normal file
View file

@ -0,0 +1,341 @@
use std::{
collections::{BTreeMap, VecDeque},
io::Read,
path::PathBuf,
};
use gix::Url;
use relative_path::RelativePathBuf;
use serde::Deserialize;
use tempfile::tempdir;
use crate::{
manifest::target::{Target, TargetKind},
names::PackageNames,
source::{
fs::{store_in_cas, FSEntry, PackageFS},
git_index::GitBasedSource,
traits::PackageSource,
version_id::VersionId,
wally::{compat_util::get_target, manifest::WallyManifest, pkg_ref::WallyPackageRef},
},
util::hash,
Project,
};
mod compat_util;
pub(crate) mod manifest;
/// The Wally package reference
pub mod pkg_ref;
/// The Wally dependency specifier
pub mod specifier;
/// The Wally package source
#[derive(Debug, Hash, PartialEq, Eq, Clone)]
pub struct WallyPackageSource {
repo_url: Url,
}
impl GitBasedSource for WallyPackageSource {
fn path(&self, project: &Project) -> PathBuf {
project
.data_dir
.join("wally_indices")
.join(hash(self.as_bytes()))
}
fn repo_url(&self) -> &Url {
&self.repo_url
}
}
impl WallyPackageSource {
/// Creates a new Wally package source
pub fn new(repo_url: Url) -> Self {
Self { repo_url }
}
fn as_bytes(&self) -> Vec<u8> {
self.repo_url.to_bstring().to_vec()
}
/// Reads the config file
pub fn config(&self, project: &Project) -> Result<WallyIndexConfig, errors::ConfigError> {
let file = self.read_file(["config.json"], project).map_err(Box::new)?;
let string = match file {
Some(s) => s,
None => {
return Err(errors::ConfigError::Missing(Box::new(
self.repo_url.clone(),
)))
}
};
serde_json::from_str(&string).map_err(Into::into)
}
}
impl PackageSource for WallyPackageSource {
type Specifier = specifier::WallyDependencySpecifier;
type Ref = WallyPackageRef;
type RefreshError = crate::source::git_index::errors::RefreshError;
type ResolveError = errors::ResolveError;
type DownloadError = errors::DownloadError;
fn refresh(&self, project: &Project) -> Result<(), Self::RefreshError> {
GitBasedSource::refresh(self, project)
}
fn resolve(
&self,
specifier: &Self::Specifier,
project: &Project,
_project_target: TargetKind,
) -> Result<crate::source::ResolveResult<Self::Ref>, Self::ResolveError> {
let (scope, name) = specifier.name.as_str();
let string = match self.read_file([scope, name], project) {
Ok(Some(s)) => s,
Ok(None) => return Err(Self::ResolveError::NotFound(specifier.name.to_string())),
Err(e) => {
return Err(Self::ResolveError::Read(
specifier.name.to_string(),
Box::new(e),
))
}
};
let entries: Vec<WallyManifest> = string
.lines()
.map(serde_json::from_str)
.collect::<Result<_, _>>()
.map_err(|e| Self::ResolveError::Parse(specifier.name.to_string(), e))?;
log::debug!("{} has {} possible entries", specifier.name, entries.len());
Ok((
PackageNames::Wally(specifier.name.clone()),
entries
.into_iter()
.filter(|manifest| specifier.version.matches(&manifest.package.version))
.map(|manifest| {
Ok((
VersionId(manifest.package.version.clone(), TargetKind::Roblox),
WallyPackageRef {
name: specifier.name.clone(),
index_url: self.repo_url.clone(),
dependencies: manifest.all_dependencies().map_err(|e| {
Self::ResolveError::AllDependencies(specifier.to_string(), e)
})?,
version: manifest.package.version,
},
))
})
.collect::<Result<_, Self::ResolveError>>()?,
))
}
fn download(
&self,
pkg_ref: &Self::Ref,
project: &Project,
reqwest: &reqwest::blocking::Client,
) -> Result<(PackageFS, Target), Self::DownloadError> {
let config = self.config(project).map_err(Box::new)?;
let index_file = project
.cas_dir
.join("wally_index")
.join(pkg_ref.name.escaped())
.join(pkg_ref.version.to_string());
let tempdir = match std::fs::read_to_string(&index_file) {
Ok(s) => {
log::debug!(
"using cached index file for package {}@{}",
pkg_ref.name,
pkg_ref.version
);
let tempdir = tempdir()?;
let fs = toml::from_str::<PackageFS>(&s)?;
fs.write_to(&tempdir, project.cas_dir(), false)?;
return Ok((fs, get_target(project, &tempdir)?));
}
Err(e) if e.kind() == std::io::ErrorKind::NotFound => tempdir()?,
Err(e) => return Err(errors::DownloadError::ReadIndex(e)),
};
let (scope, name) = pkg_ref.name.as_str();
let url = format!(
"{}/v1/package-contents/{scope}/{name}/{}",
config.api.as_str().trim_end_matches('/'),
pkg_ref.version
);
let mut response = reqwest.get(url).header(
"Wally-Version",
std::env::var("PESDE_WALLY_VERSION")
.as_deref()
.unwrap_or("0.3.2"),
);
if let Some(token) = &project.auth_config.github_token {
log::debug!("using token for wally package download");
response = response.header("Authorization", format!("Bearer {token}"));
}
let response = response.send()?.error_for_status()?;
let bytes = response.bytes()?;
let mut archive = zip::ZipArchive::new(std::io::Cursor::new(bytes))?;
archive.extract(tempdir.path())?;
let mut entries = BTreeMap::new();
let mut dir_entries = std::fs::read_dir(tempdir.path())?.collect::<VecDeque<_>>();
while let Some(entry) = dir_entries.pop_front() {
let entry = entry?;
let path =
RelativePathBuf::from_path(entry.path().strip_prefix(tempdir.path())?).unwrap();
if path == ".git" {
continue;
}
if entry.file_type()?.is_dir() {
entries.insert(path, FSEntry::Directory);
dir_entries.extend(std::fs::read_dir(entry.path())?);
continue;
}
let mut file = std::fs::File::open(entry.path())?;
let mut contents = vec![];
file.read_to_end(&mut contents)?;
let hash = store_in_cas(&project.cas_dir, &contents)?.0;
entries.insert(path, FSEntry::File(hash));
}
let fs = PackageFS(entries);
if let Some(parent) = index_file.parent() {
std::fs::create_dir_all(parent).map_err(errors::DownloadError::WriteIndex)?;
}
std::fs::write(&index_file, toml::to_string(&fs)?)
.map_err(errors::DownloadError::WriteIndex)?;
Ok((fs, get_target(project, &tempdir)?))
}
}
/// A Wally index config
#[derive(Debug, Clone, Deserialize)]
pub struct WallyIndexConfig {
api: url::Url,
}
/// Errors that can occur when interacting with a Wally package source
pub mod errors {
use thiserror::Error;
use crate::source::git_index::errors::ReadFile;
/// Errors that can occur when resolving a package from a Wally package source
#[derive(Debug, Error)]
#[non_exhaustive]
pub enum ResolveError {
/// Error interacting with the filesystem
#[error("error interacting with the filesystem")]
Io(#[from] std::io::Error),
/// Package not found in index
#[error("package {0} not found")]
NotFound(String),
/// Error reading file for package
#[error("error reading file for {0}")]
Read(String, #[source] Box<ReadFile>),
/// Error parsing file for package
#[error("error parsing file for {0}")]
Parse(String, #[source] serde_json::Error),
/// Error parsing file for package as utf8
#[error("error parsing file for {0} to utf8")]
Utf8(String, #[source] std::string::FromUtf8Error),
/// Error parsing all dependencies
#[error("error parsing all dependencies for {0}")]
AllDependencies(
String,
#[source] crate::manifest::errors::AllDependenciesError,
),
}
/// Errors that can occur when reading the config file for a Wally package source
#[derive(Debug, Error)]
#[non_exhaustive]
pub enum ConfigError {
/// Error reading file
#[error("error reading config file")]
ReadFile(#[from] Box<ReadFile>),
/// Error parsing config file
#[error("error parsing config file")]
Parse(#[from] serde_json::Error),
/// The config file is missing
#[error("missing config file for index at {0}")]
Missing(Box<gix::Url>),
}
/// Errors that can occur when downloading a package from a Wally package source
#[derive(Debug, Error)]
#[non_exhaustive]
pub enum DownloadError {
/// Error reading index file
#[error("error reading config file")]
ReadFile(#[from] Box<ConfigError>),
/// Error downloading package
#[error("error downloading package")]
Download(#[from] reqwest::Error),
/// Error deserializing index file
#[error("error deserializing index file")]
Deserialize(#[from] toml::de::Error),
/// Error reading index file
#[error("error reading index file")]
ReadIndex(#[source] std::io::Error),
/// Error decompressing archive
#[error("error decompressing archive")]
Decompress(#[from] zip::result::ZipError),
/// Error interacting with the filesystem
#[error("error interacting with the filesystem")]
Io(#[from] std::io::Error),
/// Error stripping prefix from path
#[error("error stripping prefix from path")]
StripPrefix(#[from] std::path::StripPrefixError),
/// Error serializing index file
#[error("error serializing index file")]
SerializeIndex(#[from] toml::ser::Error),
/// Error getting lib path
#[error("error getting lib path")]
LibPath(#[from] crate::source::wally::compat_util::errors::FindLibPathError),
/// Error writing index file
#[error("error writing index file")]
WriteIndex(#[source] std::io::Error),
}
}

View file

@ -0,0 +1,58 @@
use std::collections::BTreeMap;
use semver::Version;
use serde::{Deserialize, Serialize};
use crate::{
manifest::{target::TargetKind, DependencyType},
names::wally::WallyPackageName,
source::{wally::WallyPackageSource, DependencySpecifiers, PackageRef, PackageSources},
};
/// A Wally package reference
#[derive(Debug, Serialize, Deserialize, Clone, Eq, PartialEq)]
pub struct WallyPackageRef {
/// The name of the package
#[serde(rename = "wally")]
pub name: WallyPackageName,
/// The version of the package
pub version: Version,
/// The index of the package
#[serde(
serialize_with = "crate::util::serialize_gix_url",
deserialize_with = "crate::util::deserialize_gix_url"
)]
pub index_url: gix::Url,
/// The dependencies of the package
#[serde(default, skip_serializing_if = "BTreeMap::is_empty")]
pub dependencies: BTreeMap<String, (DependencySpecifiers, DependencyType)>,
}
impl PackageRef for WallyPackageRef {
fn dependencies(&self) -> &BTreeMap<String, (DependencySpecifiers, DependencyType)> {
&self.dependencies
}
fn use_new_structure(&self) -> bool {
false
}
fn target_kind(&self) -> TargetKind {
TargetKind::Roblox
}
fn source(&self) -> PackageSources {
PackageSources::Wally(WallyPackageSource::new(self.index_url.clone()))
}
}
impl Ord for WallyPackageRef {
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
self.version.cmp(&other.version)
}
}
impl PartialOrd for WallyPackageRef {
fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
Some(self.cmp(other))
}
}

View file

@ -0,0 +1,26 @@
use std::fmt::Display;
use semver::VersionReq;
use serde::{Deserialize, Serialize};
use crate::{names::wally::WallyPackageName, source::DependencySpecifier};
/// The specifier for a Wally dependency
#[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Eq, Hash)]
pub struct WallyDependencySpecifier {
/// The name of the package
#[serde(rename = "wally")]
pub name: WallyPackageName,
/// The version requirement for the package
pub version: VersionReq,
/// The index to use for the package
#[serde(default, skip_serializing_if = "Option::is_none")]
pub index: Option<String>,
}
impl DependencySpecifier for WallyDependencySpecifier {}
impl Display for WallyDependencySpecifier {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "{}@{}", self.name, self.version)
}
}