競技プログラミング用にRustコードを一つの.rs
ファイルにバンドルするCargoサブコマンドです。
```toml [package] name = "solve" version = "0.0.0" edition = "2018"
[dependencies] ac-library-rs-parted-convolution = { git = "https://github.com/qryxip/ac-library-rs-parted" } ac-library-rs-parted-dsu = { git = "https://github.com/qryxip/ac-library-rs-parted" } ac-library-rs-parted-fenwicktree = { git = "https://github.com/qryxip/ac-library-rs-parted" } ac-library-rs-parted-lazysegtree = { git = "https://github.com/qryxip/ac-library-rs-parted" } ac-library-rs-parted-math = { git = "https://github.com/qryxip/ac-library-rs-parted" } ac-library-rs-parted-maxflow = { git = "https://github.com/qryxip/ac-library-rs-parted" } ac-library-rs-parted-mincostflow = { git = "https://github.com/qryxip/ac-library-rs-parted" } ac-library-rs-parted-modint = { git = "https://github.com/qryxip/ac-library-rs-parted" } ac-library-rs-parted-scc = { git = "https://github.com/qryxip/ac-library-rs-parted" } ac-library-rs-parted-segtree = { git = "https://github.com/qryxip/ac-library-rs-parted" } ac-library-rs-parted-string = { git = "https://github.com/qryxip/ac-library-rs-parted" } ac-library-rs-parted-twosat = { git = "https://github.com/qryxip/ac-library-rs-parted" } input = { path = "/path/to/input" } output = { path = "/path/to/output" } tonellishanks = { path = "/path/to/tonellishanks" }
```
``rust
// Uncomment this line if you don't use your libraries. (
--check` still works)
//#![cfgattr(cargoequip, cargo_equip::skip)]
extern crate input as _;
use aclmodint::ModInt; use std::io::Write as _; use tonellishanks::ModIntBaseExt as _;
use permutohedron 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 --resolve-cfgs --remove docs --minify libs --rustfmt --check -o ./bundled.rs
Running `/home/ryo/.rustup/toolchains/stable-x86_64-unknown-linux-gnu/bin/cargo check --message-format json -p -p 'ac-library-rs-parted-build:0.1.0' -p 'ac-library-rs-parted-convolution:0.1.0' -p 'ac-library-rs-parted-dsu:0.1.0' -p 'ac-library-rs-parted-fenwicktree:0.1.0' -p 'ac-library-rs-parted-internal-bit:0.1.0' -p 'ac-library-rs-parted-internal-math:0.1.0' -p 'ac-library-rs-parted-internal-queue:0.1.0' -p 'ac-library-rs-parted-internal-scc:0.1.0' -p 'ac-library-rs-parted-internal-type-traits:0.1.0' -p 'ac-library-rs-parted-lazysegtree:0.1.0' -p 'ac-library-rs-parted-math:0.1.0' -p 'ac-library-rs-parted-maxflow:0.1.0' -p 'ac-library-rs-parted-mincostflow:0.1.0' -p 'ac-library-rs-parted-modint:0.1.0' -p 'ac-library-rs-parted-scc:0.1.0' -p 'ac-library-rs-parted-segtree:0.1.0' -p 'ac-library-rs-parted-string:0.1.0' -p 'ac-library-rs-parted-twosat:0.1.0' -p 'anyhow:1.0.34' -p 'byteorder:1.3.4' -p 'num-traits:0.2.14' -p 'proc-macro2:1.0.10' -p 'ryu:1.0.5' -p 'serde:1.0.113' -p 'serde_derive:1.0.113' -p 'serde_json:1.0.59' -p 'syn:1.0.17' -p 'typenum:1.12.0'`
Finished dev [unoptimized + debuginfo] target(s) in 0.03s
Running `/home/ryo/.cargo/bin/rustup run nightly cargo udeps --output json -p solve --bin solve`
Checking solve v0.0.0 (/home/ryo/src/local/a/solve)
Finished dev [unoptimized + debuginfo] target(s) in 0.16s
info: Loading save analysis from "/home/ryo/src/local/a/solve/target/debug/deps/save-analysis/solve-2970d6e10b9c0877.json"
Bundling the code
Checking cargo-equip-check-output-nq4nm7zkj9vtgbd9 v0.1.0 (/tmp/cargo-equip-check-output-nq4nm7zkj9vtgbd9)
Finished dev [unoptimized + debuginfo] target(s) in 0.35s
Submit Info #29083 - Library-Checker
nightly
ツールチェインとcargo-udepsもインストールしてください。
console
❯ rustup update nightly
console
❯ cargo install --git https://github.com/est31/cargo-udeps # for est31/cargo-udeps#80
console
❯ cargo install cargo-equip
master
console
❯ cargo install --git https://github.com/qryxip/cargo-equip
バイナリでの提供もしています。
cargo-equip
で展開できるライブラリには以下の制約があります。
各lib
には#[macro_export]
したマクロと同名なアイテムが存在しないようにする。
cargo-equipはmod lib_name
直下にpub use crate::{ それらの名前 }
を挿入するため、展開後のuse
で壊れます。
マクロは#[macro_use]
で使ってください。
```rust // in main source code
extern crate input as _; ```
展開時にはコメントアウトされます。
```rust // in main source code
/#[macro_use]
extern crate input as _;/ // as _
でなければuse crate::$name;
が挿入される
```
共に展開する予定のクレートを使う場合、extern crate
を宣言してそれを適当な場所にマウントし、そこを相対パスで参照する。
cargo-equipはitertools
等のAtCoderやCodinGameで使えるクレートを除いて、
extern crate
をuse crate::extern_crate_name_in_main_crate;
に置き換えます。
誤って直接使わないようにlib
→ lib
の依存においては対象の名前はリネームしておくことを強く推奨します。
rust
extern crate __another_lib as another_lib;
注意点として、バンドル後のコードが2015 editionで実行される場合(AOJ, 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;
またマクロ内では$crate
を使ってください。
macro_rules!
内の$crate
は$crate::extern_crate_name_in_main_crate
に置き換えられます。
可能な限りライブラリを小さなクレートに分割する。
cargo-equipは「クレート内のアイテムの依存関係」を調べることはしません。 出力結果を64KiBに収めるためにはできるだけ小さなクレートに分割してください。
console
.
├── input
│ ├── Cargo.lock
│ ├── Cargo.toml
│ └── src
│ └── lib.rs
├── output
│ ├── Cargo.lock
│ ├── Cargo.toml
│ └── src
│ └── lib.rs
⋮
ライブラリが用意できたら、それらをbin
側のCargo.toml
の[dependencies]
に加えてください。
コンテスト毎にツールでパッケージを自動生成しているならそれのテンプレートに加えてください。
ac-library-rsを使いたい場合、ac-library-rs-partedを使ってください。
toml
[dependencies]
ac-library-rs-parted-convolution = { git = "https://github.com/qryxip/ac-library-rs-parted" }
ac-library-rs-parted-dsu = { git = "https://github.com/qryxip/ac-library-rs-parted" }
ac-library-rs-parted-fenwicktree = { git = "https://github.com/qryxip/ac-library-rs-parted" }
ac-library-rs-parted-lazysegtree = { git = "https://github.com/qryxip/ac-library-rs-parted" }
ac-library-rs-parted-math = { git = "https://github.com/qryxip/ac-library-rs-parted" }
ac-library-rs-parted-maxflow = { git = "https://github.com/qryxip/ac-library-rs-parted" }
ac-library-rs-parted-mincostflow = { git = "https://github.com/qryxip/ac-library-rs-parted" }
ac-library-rs-parted-modint = { git = "https://github.com/qryxip/ac-library-rs-parted" }
ac-library-rs-parted-scc = { git = "https://github.com/qryxip/ac-library-rs-parted" }
ac-library-rs-parted-segtree = { git = "https://github.com/qryxip/ac-library-rs-parted" }
ac-library-rs-parted-string = { git = "https://github.com/qryxip/ac-library-rs-parted" }
ac-library-rs-parted-twosat = { git = "https://github.com/qryxip/ac-library-rs-parted" }
準備ができたらコードを書いてください。
bin
側の制約は以下の2つです。
use
しない。qualified pathで使うか#[macro_use]
で使う。bin
内にmod
を作る場合、その中ではExtern Preludeから名前を解決しない。``rust
// Uncomment this line if you don't use your libraries. (
--check` still works)
//#![cfgattr(cargoequip, cargo_equip::skip)]
extern crate input as _;
use std::io::Write as _;
fn main() { input! { n: usize, }
output::buf_print(|out| {
macro_rules! println(($($tt:tt)*) => (writeln!(out, $($tt)*).unwrap()));
for i in 1..=n {
match i % 15 {
0 => println!("Fizz Buzz"),
3 | 6 | 9 | 12 => println!("Fizz"),
5 | 10 => println!("Buzz"),
_ => println!("{}", i),
}
}
});
} ```
コードが書けたらcargo equip
で展開します。
--bin {binの名前}
か--src {binのファイルパス}
でbin
を指定してください。
パッケージ内のbin
が一つの場合は省略できます。
ただしdefault-run
には未対応です。
console
❯ cargo equip --bin "$name"
コードはこのように展開されます。
extern_crate_name
がbin
側から与えられていないクレートは"__internal_lib_0_1_0" + &"_".repeat(n)
のような名前が与えられます。
``diff
+//! # Bundled libraries
+//!
+//! ##
input(private)
+//!
+//! ###
externcratename
+//!
+//!
input
+//!
+//! ##
output(private)
+//!
+//! ###
externcratename
+//!
+//!
output`
// Uncomment this line if you don't use your libraries. (--check
still works)
//#![cfgattr(cargoequip, cargo_equip::skip)]
-#[macrouse] -extern crate input as _; +/*#[macrouse] +extern crate input as _;*/
use std::io::Write as _;
fn main() { input! { n: usize, }
output::buf_print(|out| {
macro_rules! println(($($tt:tt)*) => (writeln!(out, $($tt)*).unwrap()));
for i in 1..=n {
match i % 15 {
0 => println!("Fizz Buzz"),
3 | 6 | 9 | 12 => println!("Fizz"),
5 | 10 => println!("Buzz"),
_ => println!("{}", i),
}
}
});
}
+
+// The following code was expanded by cargo-equip
.
+
+#[allow(deadcode)]
+mod input {
+ // ...
+}
+
+#[allow(deadcode)]
+mod output {
+ // ...
+}
```
cargo-equipがやる操作は以下の通りです。
bin
側
#![cfg_attr(cargo_equip, cargo_equip::skip)]
を発見した場合、以下の処理をスキップして--check
の処理だけ行い出力mod $name;
をすべて再帰的に展開する。このとき各モジュールをインデントする。ただし複数行にまたがるリテラルが無い場合はインデントしないextern crate
を処理lib
を下部に追加lib
側
mod $name;
をすべて再帰的に展開するcrate
を処理extern crate
を処理macro_rules!
を処理--resolve-cfg
オプションを付けた場合、#[cfg(常にTRUEのように見える式)]
のアトリビュートと#[cfg(常にFALSEのように見える式)]
のアトリビュートが付いたアイテムを消去--remove docs
オプションを付けた場合、doc commentを消去--remove comments
オプションを付けた場合、commentを消去--minify all
オプションを付けた場合コード全体を最小化する--rustfmt
オプションを付けた場合Rustfmtでフォーマットする--resolve-cfgs
#[cfg(常にTRUEのように見える式)]
(e.g. cfg(feature = "enabled-feature")
)のアトリビュートを消去します。#[cfg(常にFALSEのように見える式)]
(e.g. cfg(test)
, cfg(feature = "disable-feature")
)のアトリビュートが付いたアイテムを消去します。```rust
pub mod a { pub struct A;
#[cfg(test)]
mod tests {
#[test]
fn it_works() {
assert_eq!(2 + 2, 4);
}
}
} ```
↓
```rust
pub mod a { pub struct A; } ```
--remove <REMOVE>...
--remove docs
でDoc comment (//! ..
, /// ..
, /** .. */
, #[doc = ".."]
)を--remove comments
でコメント (// ..
, /* .. */
)を除去します。
```rust
pub mod a { //! A.
/// A.
pub struct A; // aaaaa
} ```
↓
```rust
pub mod a { pub struct A; } ```
--minify <MINIFY>
--minify lib
で展開後のライブラリをそれぞれ一行に折り畳みます。
--minify all
でコード全体を最小化します。
ただ現段階では実装が適当なのでいくつか余計なスペースが挟まる場合があります。
--rustfmt
出力をRustfmtでフォーマットします。
--check
バンドルしたコードを出力する前にtarget directoryを共有した一時パッケージを作り、それの上でcargo check
します。
#![cfg_attr(cargo_equip, cargo_equip::skip)]
でスキップした場合もチェックします。
console
❯ cargo equip --check -o /dev/null
Running `/home/ryo/.cargo/bin/rustup run nightly cargo udeps --output json -p solve --bin solve`
Checking solve v0.0.0 (/home/ryo/src/local/a/solve)
Finished dev [unoptimized + debuginfo] target(s) in 0.13s
info: Loading save analysis from "/home/ryo/src/local/a/solve/target/debug/deps/save-analysis/solve-4eea33c8603d6001.json"
Bundling the code
Checking cargo-equip-check-output-6j2i3j3tgtugeaqm v0.1.0 (/tmp/cargo-equip-check-output-6j2i3j3tgtugeaqm)
Finished dev [unoptimized + debuginfo] target(s) in 0.11s
MIT or Apache-2.0のデュアルライセンスです。