競技プログラミング用にRustコードを一つの.rs
ファイルにバンドルするCargoサブコマンドです。
```toml [package] name = "my_library" version = "0.0.0" edition = "2018"
[package.metadata.cargo-equip.module-dependencies] "crate::input" = [] "crate::output" = [] "crate::tonellishanks" = ["crate::xorshift", "::_aclrs::modint"] "crate::xorshift" = []
"::aclrs::convolution" = ["::aclrs::internalbit", "::aclrs::modint"] "::aclrs::internalbit" = [] "::aclrs::internalmath" = [] "::aclrs::internalqueue" = [] "::aclrs::internalscc" = [] "::aclrs::internaltypetraits" = [] "::aclrs::lazysegtree" = ["::aclrs::internalbit", "::aclrs::segtree"] "::aclrs::math" = ["::aclrs::internalmath"] "::aclrs::maxflow" = ["::aclrs::internaltypetraits", "::aclrs::internalqueue"] "::aclrs::mincostflow" = ["::aclrs::internaltypetraits"] "::aclrs::modint" = ["::aclrs::internalmath"] "::aclrs::scc" = ["::aclrs::internalscc"] "::aclrs::segtree" = ["::aclrs::internalbit", "::aclrs::internaltypetraits"] "::aclrs::twosat" = ["::aclrs::internalscc"]
[dependencies] __aclrs = { package = "ac-library-rs", git = "https://github.com/rust-lang-ja/ac-library-rs", branch = "replace-absolute-paths" } ```
```toml [package] name = "solve" version = "0.0.0" edition = "2018"
[dependencies] _aclrs = { package = "ac-library-rs", git = "https://github.com/rust-lang-ja/ac-library-rs", branch = "replace-absolute-paths" } _my = { package = "mylibrary", path = "../mylibrary" } ```
```rust
use ::aclrs::modint::ModInt; use ::my::{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 libs --rustfmt --check -o ./bundled.rs
Bundling the code
warning: found `crate` paths. replacing them with `crate::__aclrs`
warning: could not minify the code. inserting spaces: `crate::__aclrs`
Checking cargo-equip-check-output-ea0pb2d6u0yda27w v0.1.0 (/tmp/cargo-equip-check-output-ea0pb2d6u0yda27w)
Finished dev [unoptimized + debuginfo] target(s) in 0.32s
Submit Info #27189 - Library-Checker
console
$ cargo install cargo-equip
master
console
$ cargo install --git https://github.com/qryxip/cargo-equip
バイナリでの提供もしています。
cargo-equip
で展開できるライブラリには以下の制約があります。
#[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している場合は一切操作しません。
```diff #[cfgattr(cargoequip, cargoequip::translatedollarcrates)] #[macroexport] macro_rules! input { ($($tt:tt)*) => {
ただ1.の制約を守っていて、関連するアイテムがすべて同じモジュールにあるなら無くても動く場合があります。
共に展開する予定のクレートを使う場合、各モジュールに#[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;
可能な限り絶対パスを使わない。
cargo-equipはpathのcrate
はcrate::extern_crate_name_in_main_crate
に、pub(crate)
はpub(in crate::extern_crate_name_in_main_crate)
に置き換えます。
ただしこの置き換えは必ず上手くいくかどうかがわかりません。
できる限りcrate::
よりもself::
とsuper::
を使ってください。
diff
-use crate::foo::Foo;
+use super::foo::Foo;
可能な限りinline module (mod $name;
)は深さ1まで
後述するpackage.metadata
でのモジュール依存関係の記述は今のところトップモジュール単位です。
crate root直下にはmod
以外のpub
なアイテムが置かれていない
置いてもいいですが使わないでください。
現在cargo-equipはlib.rs
直下のmod
以外のアイテムを、読みますがすべて消し飛ばします。
5.と6.はそのうち対応しようと思います。
このように薄く広く作ってください。
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
__aclrs
の場合[package.metadata.cargo-equip.module-dependencies] "::aclrs::convolution" = ["::aclrs::internalbit", "::aclrs::modint"] "::aclrs::internalbit" = [] "::aclrs::internalmath" = [] "::aclrs::internalqueue" = [] "::aclrs::internalscc" = [] "::aclrs::internaltypetraits" = [] "::aclrs::lazysegtree" = ["::aclrs::internalbit", "::aclrs::segtree"] "::aclrs::math" = ["::aclrs::internalmath"] "::aclrs::maxflow" = ["::aclrs::internaltypetraits", "::aclrs::internalqueue"] "::aclrs::mincostflow" = ["::aclrs::internaltypetraits"] "::aclrs::modint" = ["::aclrs::internalmath"] "::aclrs::scc" = ["::aclrs::internalscc"] "::aclrs::segtree" = ["::aclrs::internalbit", "::aclrs::internaltypetraits"] "::aclrs::twosat" = ["::aclrs::internalscc"] ```
そしてbin
側の準備として、バンドルしたいライブラリをCargo.toml
のdependencies
に加えてください。
ただしこの際、ライブラリは誤って直接使わないようにリネームしておくことを強く推奨します。
直接使った場合cargo-equip
はそれについて何も操作しません。
コンテスト毎にツールでパッケージを自動生成しているならそれのテンプレートに加えてください。
toml
[dependencies]
__aclrs = { package = "ac-library-rs", git = "https://github.com/rust-lang-ja/ac-library-rs", branch = "replace-absolute-paths" }
__my_lib1 = { package = "my_lib1", path = "/path/to/my_lib1" }
__my_lib2 = { package = "my_lib2", path = "/path/to/my_lib2" }
準備ができたらこのようにattribute付きでライブラリをuse
します。
```rust
use ::mylib1::{a::A, b::B}; use ::mylib2::{c::C, d::D}; ```
leading colon (::
)が付いたuse
だけ展開されます。
use ::__my_lib1::{a::A, b::B};
^^
パスの1つ目のsegmentから展開するべきライブラリを決定します。 leading colonを必須としているのはこのためです。
use ::__my_lib1::{a::A, b::B};
^^^^^^^^^
先述したライブラリの制約より、パスの第二セグメントはモジュールとみなします。
これらのモジュールと、先程書いたmodule-dependencies
で繋がっているモジュールが展開されます。
use ::__my_lib1::{a::A, b::B};
^ ^
第三セグメント以降はuse self::$extern_crate_name::$module_name::{..}
と展開されます。
use ::__my_lib1::{a::A, b::B};
^ ^
コードが書けたらcargo equip
で展開します。
--bin {binの名前}
か--src {binのファイルパス}
でbin
を指定してください。
パッケージ内のbin
が一つの場合は省略できます。
ただしdefault-run
には未対応です。
console
$ cargo equip --bin "$name"
コードはこのように展開されます。
``rust
//! # Bundled libraries
//!
//! ## [
mylib1]({ a link to Crates.io or the repository })
//!
//! ### Modules
//!
//! -
::mylib1::a
//! -
::mylib1::b
//!
//! ## [
mylib2]({ a link to Crates.io or the repository })
//!
//! ### Modules
//!
//! -
::mylib2::c
//! -
::mylib2::d`
/#![cfg_attr(cargo_equip, cargo_equip::equip)]/
/use ::my_lib1::{a::A, b::B};/ /use ::my_lib2::{c::C, d::D};/
fn main() { todo!(); }
// The following code was expanded by cargo-equip
.
use self::mylib1::{a::A, b::B}; use self::mylib2::{c::C, d::D};
pub mod _mylib { mod a { // .. }
mod b {
// ..
}
// `module-dependencies`で連結されているモジュールも共に展開される
mod b_dep {
// ..
}
}
pub mod _mylib { mod c { // .. }
mod d {
// ..
}
} ```
cargo-equipがやる操作は以下の通りです。これ以外は何も行いません。
bin
側
#![cfg_attr(cargo_equip, cargo_equip::equip)]
を検知し、コメントアウト::
で始まるuse
を検知し、コメントアウトlib
を下部に追加lib
側
lib.rs
内のmod
をすべて再帰的に展開する。このとき各モジュールをインデントする。ただし複数行にまたがるリテラルが無い場合はインデントしないmod
を展開後、mod
とextern crate
以外のすべてのトップレベルのアイテムを消去する#[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 lib
で展開後のライブラリをそれぞれ一行に折り畳みます。
--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のデュアルライセンスです。