競技プログラミング用にRustコードを一つの.rs
ファイルにバンドルするCargoサブコマンドです。
```toml [package] name = "lib" version = "0.0.0" edition = "2018"
[package.metadata.cargo-equip.module-dependencies] "crate::input" = [] "crate::output" = [] "crate::tonellishanks" = ["crate::xorshift", "::_atcoder::modint"] "crate::xorshift" = []
"::atcoder::convolution" = ["::atcoder::internalbit", "::atcoder::modint"] "::atcoder::internalbit" = [] "::atcoder::internalmath" = [] "::atcoder::internalqueue" = [] "::atcoder::internalscc" = [] "::atcoder::internaltypetraits" = [] "::atcoder::lazysegtree" = ["::atcoder::internalbit", "::atcoder::segtree"] "::atcoder::math" = ["::atcoder::internalmath"] "::atcoder::maxflow" = ["::atcoder::internaltypetraits", "::atcoder::internalqueue"] "::atcoder::mincostflow" = ["::atcoder::internaltypetraits"] "::atcoder::modint" = ["::atcoder::internalmath"] "::atcoder::scc" = ["::atcoder::internalscc"] "::atcoder::segtree" = ["::atcoder::internalbit", "::atcoder::internaltypetraits"] "::atcoder::twosat" = ["::atcoder::internalscc"]
[dependencies] __atcoder = { package = "ac-library-rs", git = "https://github.com/rust-lang-ja/ac-library-rs", branch = "replace-absolute-paths" } ```
```toml [package] name = "bin" version = "0.0.0" edition = "2018"
[dependencies] _atcoder = { package = "ac-library-rs", git = "https://github.com/rust-lang-ja/ac-library-rs", branch = "replace-absolute-paths" } _lib = { package = "lib", path = "/path/to/lib" } ```
```rust
use ::{ _atcoder::modint::ModInt, _lib::{input, output, tonelli_shanks::ModIntBaseExt as _}, };
use std::io::Write as _;
fn main() { input! { yps: [(u32, u32)], }
output::buf_print(|out| {
macro_rules! println(($($tt:tt)*) => (writeln!(out, $($tt)*).unwrap()));
for (y, p) in yps {
ModInt::set_modulus(p);
if let Some(sqrt) = ModInt::new(y).sqrt() {
println!("{}", sqrt);
} else {
println!("-1");
}
}
});
} ```
↓
console
$ cargo equip --remove docs test-items --minify mods --rustfmt --check -o ./bundled.rs
Bundling code
warning: could not minify the code. inserting spaces: `internal_math`
Checking cargo-equip-check-output-rml3nu3kghlx3ar4 v0.1.0 (/tmp/cargo-equip-check-output-rml3nu3kghlx3ar4)
Finished dev [unoptimized + debuginfo] target(s) in 0.30s
Submit Info #24741 - Library-Checker
console
$ cargo install cargo-equip
master
console
$ cargo install --git https://github.com/qryxip/cargo-equip
バイナリでの提供もしています。
cargo-equip
で展開できるライブラリには以下の制約があります。
絶対パスを使わない。クレート内のアイテムはすべて相対パスで書く。
Rustのパス解決をシミュレートするのは非常に困難であるため、cargo-equipは基本的にパスの置き換えを行いません。
crate::
はself::
とsuper::
で書き直してください。
diff
-use crate::foo::Foo;
+use super::foo::Foo;
共に展開する予定のクレートを使う場合、各モジュールに#[cfg_attr(cargo_equip, cargo_equip::use_another_lib)]
を付けたextern crate
を宣言してマウントし、そこを相対パスで参照する。
cargo-equipはこのアトリビュートの付いたextern crate
をuse crate::extern_crate_name_in_main_crate;
に置き換えます。
誤って直接使わないように対象の名前はリネームしておくことを強く推奨します。
```rust
extern crate _anotherlib as another_lib; ```
注意点として、バンドル後のコードが2015 editionで実行される場合(yukicoder, Library-Checker)、相対パスで参照するときはself::
を付けてください。
rust
use self::another_lib::foo::Foo;
#[macro_export] macro_rules! name { .. }
はmod name
の中に置かれている
cargo-equipはmain source file内でどのマクロが使われているのかを調べません。
この制約を守れば#[macro_export]
したマクロは展開前も展開後も自然に動作します。
```rust
use ::_mylib::input;
fn main() { input! { a: u32, b: u32, s: String, }
todo!();
} ```
マクロにおいて$crate::
でアイテムのパスを指定している場合、#[cfg_attr(cargo_equip, cargo_equip::translate_dollar_crates)]
を付ける
cargo-equipはこのアトリビュートが付いたmacro_rules!
内にあるすべての$crate
を、::identifier!
と続く場合のみを除いて$crate::extern_crate_name_in_the_main_crate
と置き換えます。
アトリビュートが無い場合、またはアトリビュートをtypoしている場合は一切操作しません。
ただ3.の制約を守っていて、関連するアイテムがすべて同じモジュールにあるなら無くても動く場合があります。
```diff #[cfgattr(cargoequip, cargoequip::translatedollarcrates)] #[macroexport] macro_rules! input { ($($tt:tt)*) => {
非inline module (mod $name;
)は深さ1まで
深さ2以上のモジュールはすべてinline module (mod $name { .. }
)
crate root直下にはmod
以外のpub
なアイテムが置かれていない
置いてもいいですが使わないでください。
現在cargo-equipはlib.rs
直下のmod
以外のアイテムをすべて無視します。
5.と6.と7.はそのうち対応しようと思います。
このように薄く広く作ってください。
src
├── a.rs
├── b.rs
├── c.rs
└── lib.rs
rust
// src/lib.rs
pub mod a;
pub mod b;
pub mod c;
この制限に合うようなライブラリを書いたら、そのCargo.toml
のpackage.metadata
にモジュールの依存関係を手で書いてください。
直接use
したモジュールと、その連結成分だけを展開します。
書いていない場合や欠けている場合はwarningと共にすべてのモジュールを展開します。
toml
[package.metadata.cargo-equip.module-dependencies]
"crate::a" = []
"crate::b" = ["crate::b"]
"crate::c" = ["::__another_lib::d"]
"::__another_lib::d" = ["::__another_lib::e"]
最後にマージされるのでどのモジュールの依存関係を、bin
側を含めてどのパッケージに記述してもよいです。
ac-library-rsを使う場合、毎回使用するパッケージに以下のものを加えてください。
```toml
__atcoder
の場合[package.metadata.cargo-equip.module-dependencies] "::atcoder::convolution" = ["::atcoder::internalbit", "::atcoder::modint"] "::atcoder::internalbit" = [] "::atcoder::internalmath" = [] "::atcoder::internalqueue" = [] "::atcoder::internalscc" = [] "::atcoder::internaltypetraits" = [] "::atcoder::lazysegtree" = ["::atcoder::internalbit", "::atcoder::segtree"] "::atcoder::math" = ["::atcoder::internalmath"] "::atcoder::maxflow" = ["::atcoder::internaltypetraits", "::atcoder::internalqueue"] "::atcoder::mincostflow" = ["::atcoder::internaltypetraits"] "::atcoder::modint" = ["::atcoder::internalmath"] "::atcoder::scc" = ["::atcoder::internalscc"] "::atcoder::segtree" = ["::atcoder::internalbit", "::atcoder::internaltypetraits"] "::atcoder::twosat" = ["::atcoder::internalscc"] ```
そしてbin
側の準備として、バンドルしたいライブラリをCargo.toml
のdependencies
に加えてください。
ただしこの際、ライブラリは誤って直接使わないようにリネームしておくことを強く推奨します。
直接使った場合cargo-equip
はそれについて何も操作しません。
コンテスト毎にツールでパッケージを自動生成しているならそれのテンプレートに加えてください。
toml
[dependencies]
__atcoder = { package = "ac-library-rs", git = "https://github.com/rust-lang-ja/ac-library-rs", branch = "replace-absolute-paths" }
__my_lib = { package = "my_lib", path = "/path/to/my_lib" }
準備ができたらこのようにattribute付きでライブラリをuse
します。
```rust
use ::_mylib::{a::A, b::B};
// or // //#[cfgattr(cargoequip, cargoequip::equip)] //use ::{ // _mylib1::{a::A, b::B}, // _my_lib2::{c::C, c::C}, //}; ```
use
のパスにはleading colon (::
)を付けてください。
```
use ::_mylib::{a::A, b::B}; ^^ ```
パスの1つ目のsegmentから展開するべきライブラリを決定します。 leading colonを必須としているのはこのためです。
```
use ::_mylib::{a::A, b::B}; ^^^^^^^^ ```
先述したライブラリの制約より、パスの第二セグメントはモジュールとみなします。
これらのモジュールと、先程書いたmodule-dependencies
で繋がっているモジュールが展開されます。
```
use ::_mylib::{a::A, b::B}; ^ ^ ```
第三セグメント以降はuse self::$extern_crate_name::$module_name::{..}
と展開されます。
```
use ::_mylib::{a::A, b::B}; ^ ^ ```
コードが書けたらcargo equip
で展開します。
--bin {binの名前}
か--src {binのファイルパス}
でbin
を指定してください。
パッケージ内のbin
が一つの場合は省略できます。
ただしdefault-run
には未対応です。
console
$ cargo equip --bin "$name"
コードはこのように展開されます。
``rust
//! # Bundled libraries
//!
//! ## [
mylib]({ a link to Crates.io or the repository })
//!
//! ### Modules
//!
//! -
::mylib::a
//! -
::mylib::b
//! -
::mylib::c`
/#[cfg_attr(cargo_equip, cargo_equip::equip)] use ::__my_lib::{a::A, b::B};/
fn main() { todo!(); }
// The following code was expanded by cargo-equip
.
use self::mylib::a::A; use self::mylib::b::B;
pub mod _mylib { mod a { // .. }
mod b {
// ..
}
// `a`で使われていると`mod-dependencies`に記述されているため、展開される
mod c {
// ..
}
} ```
cargo-equipがやる操作は以下の通りです。これ以外は何も行いません。
bin
側
#[cfg_attr(cargo_equip, cargo_equip::equip)]
が付いたuse
をコメントアウトlib
を下部に追加lib
側
#[cfg_attr(cargo_equip, cargo_equip::use_another_lib)]
が付いたextern crate
を操作#[cfg_attr(cargo_equip, cargo_equip::translate_dollar_crates)]
が付いたmacro_rules!
を操作--remove <REMOVE>...
オプションを付けた場合対象を操作--minify all
オプションを付けた場合コード全体を最小化する--rustfmt
オプションを付けた場合Rustfmtでフォーマットする--remove <REMOVE>...
--remove test-items
で#[cfg(test)]
が付いたアイテムを--remove docs
でDoc comment (//! ..
, /// ..
, /** .. */
, #[doc = ".."]
)を--remove comments
でコメント (// ..
, /* .. */
)を除去することができます。
```rust
pub mod a { //! A.
/// A.
pub struct A; // aaaaa
#[cfg(test)]
mod tests {
#[test]
fn it_works() {
assert_eq!(2 + 2, 4);
}
}
} ```
↓
```rust
pub mod a { pub struct A; } ```
--minify <MINIFY>
--minify mods
で展開後の各モジュールをそれぞれ一行に折り畳みます。
--minify all
でコード全体を最小化します。
ただ現段階では実装が適当なのでいくつか余計なスペースが挟まる場合があります。
--rustfmt
出力をRustfmtでフォーマットします。
--check
バンドルしたコードを出力する前にtarget directoryを共有した一時パッケージを作り、それの上でcargo check
します。
console
$ cargo equip --check -o /dev/null
Bundling code
Checking cargo-equip-check-output-r3cw9cy0swqb5yac v0.1.0 (/tmp/cargo-equip-check-output-r3cw9cy0swqb5yac)
Finished dev [unoptimized + debuginfo] target(s) in 0.18s
MIT or Apache-2.0のデュアルライセンスです。