cargo-equip

CI codecov unsafe forbidden Crates.io Crates.io

English

競技プログラミング用にRustコードを一つの.rsファイルにバンドルするCargoサブコマンドです。

Sqrt Mod - Library-Cheker

```toml [package] name = "bin" version = "0.0.0" authors = ["Ryo Yamashita qryxip@gmail.com"] edition = "2018" publish = false

[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"] "::lib::input" = [] "::lib::output" = [] "::lib::tonelli_shanks" = ["::lib::xorshift"] "::lib::xorshift" = []

..

[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

[cfgattr(cargoequip, cargo_equip::equip)]

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

インストール

Crates.io

console $ cargo install cargo-equip

master

console $ cargo install --git https://github.com/qryxip/cargo-equip

GitHub Releases

バイナリでの提供もしています。

使い方

cargo-equipで展開できるライブラリには以下の制限があります。

  1. 絶対パスを使わない。クレート内のアイテムはすべて相対パスで書く。

    Rustのパス解決をシミュレートするのは非常に困難であるため、cargo-equipはパスの置き換えを行いません。 crate::self::super::で書き直してください。

    diff -use crate::foo::Foo; +use super::foo::Foo;

  2. 共に展開する予定のクレートを使う場合、各モジュールに#[cfg_attr(cargo_equip, cargo_equip::use_another_lib)]を付けたextern crateを宣言してマウントし、そこを相対パスで参照する。

    cargo-equipはこのアトリビュートの付いたextern crateuse crate::extern_crate_name_in_main_crate;に置き換えます。

    誤って直接使わないように対象の名前はリネームしておくことを強く推奨します。

    ```rust

    [cfgattr(cargoequip, cargoequip::useanother_lib)]

    extern crate _anotherlib as another_lib; ```

    注意点として、バンドル後のコードが2015 editionで実行される場合(yukicoder, Library-Checker)、相対パスで参照するときはself::を付けてください。

    diff -use another_lib::Foo; +use self::another_lib::Foo;

  3. #[macro_export] macro_rules! name { .. }mod nameの中に置かれている

    cargo-equipはmain source file内でどのマクロが使われているのかを調べません。 この制約を守れば#[macro_export]したマクロは展開前も展開後も自然に動作します。

    ```rust

    [cfgattr(cargoequip, cargo_equip::equip)]

    use ::_mylib::input; ```

  4. マクロにおいて$crate::でアイテムのパスを指定している場合、#[cfg_attr(cargo_equip, cargo_equip::translate_dollar_crates)]を付ける

    cargo-equipはこのアトリビュートが付いたmacro_rules!内にあるすべての$crateを、::identifier!と続く場合のみを除いて$crate::extern_crate_nameと置き換えます。 アトリビュートが無い場合、またはアトリビュートをtypoしている場合は一切操作しません。 4.の制約を守っているなら無くても動く場合があります。

    ```rust

    [cfgattr(cargoequip, cargoequip::translatedollar_crates)]

    [macro_export]

    macro_rules! input { () => { // .. }; } ```

  5. 非inline module (mod $name;)は深さ1まで

  6. 深さ2以上のモジュールはすべてinline module (mod $name { .. })

  7. 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;

そしてbin側の準備として、バンドルしたいライブラリをCargo.tomldependenciesに加えてください。 コンテスト毎にツールでパッケージを自動生成しているならそれのテンプレートに加えてください。

ただしこの際、ライブラリは誤って直接使わないようにリネームしておくことを強く推奨します。 直接使った場合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" }

この制限に合うようなライブラリを書いたら、そのCargo.tomlpackage.metadataにモジュールの依存関係を手で書いてください。 直接useしたモジュールと、その連結成分だけを展開します。 書いていない場合や欠けている場合はwarningと共にすべてのモジュールを展開します。

toml [package.metadata.cargo-equip.module-dependencies] "::__my_lib::a" = ["::__my_lib::c"] "::__my_lib::b" = [] "::__my_lib::c" = []

準備ができたらこのようにattribute付きでライブラリをuseします。

```rust

[cfgattr(cargoequip, cargo_equip::equip)]

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 (::)を付けてください。

```

[cfgattr(cargoequip, cargo_equip::equip)]

use ::_mylib::{a::A, b::B}; ^^ ```

パスの1つ目のsegmentから展開するべきライブラリを決定します。 leading colonを必須としているのはこのためです。

```

[cfgattr(cargoequip, cargo_equip::equip)]

use ::_mylib::{a::A, b::B}; ^^^^^^^^ ```

先述したライブラリの制約より、パスの第二セグメントはモジュールとみなします。 これらのモジュールと、先程書いたmodule-dependenciesで繋がっているモジュールが展開されます。

```

[cfgattr(cargoequip, cargo_equip::equip)]

use ::_mylib::{a::A, b::B}; ^ ^ ```

第三セグメント以降はuse self::$extern_crate_name::$module_name::{..}と展開されます。

```

[cfgattr(cargoequip, cargo_equip::equip)]

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;

[allow(dead_code)]

pub mod _mylib { mod a { // .. }

mod b {
    // ..
}

// `a`で使われていると`mod-dependencies`に記述されているため、展開される
mod c {
    // ..
}

} ```

cargo-equipがやる操作は以下の通りです。これ以外は何も行いません。

オプション

--remove <REMOVE>...

  1. --remove test-items#[cfg(test)]が付いたアイテムを
  2. --remove docsでDoc comment (//! .., /// .., /** .. */, #[doc = ".."])を
  3. --remove commentsでコメント (// .., /* .. */)を

除去することができます。

```rust

[allow(dead_code)]

pub mod a { //! A.

/// A.
pub struct A; // aaaaa

#[cfg(test)]
mod tests {
    #[test]
    fn it_works() {
        assert_eq!(2 + 2, 4);
    }
}

} ```

```rust

[allow(dead_code)]

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のデュアルライセンスです。