2023-02-21 12:43:43 +01:00
use std ::env ;
use std ::fs ;
use std ::io ::Write ;
use std ::path ::{ Path , PathBuf } ;
use clap_complete ::Shell ;
use anyhow ::{ Context , Result } ;
2023-10-20 13:04:00 +02:00
pub mod cli {
include! ( " src/cli/mod.rs " ) ;
2023-02-21 12:43:43 +01:00
}
fn main ( ) {
println! ( " cargo:rerun-if-changed=build.rs " ) ;
// Generate subplot tests.
2023-04-15 12:04:53 +02:00
#[ cfg(feature = " subplot " ) ]
2023-02-21 12:43:43 +01:00
subplot_build ::codegen ( Path ::new ( " sq.subplot " ) )
. expect ( " failed to generate code with Subplot " ) ;
2023-10-20 13:04:00 +02:00
let mut sq = cli ::build ( ) ;
2023-02-21 12:43:43 +01:00
// Dump help output of all commands and subcommands, for inclusion in docs
dump_help ( sq . clone ( ) ) . unwrap ( ) ;
// Generate shell completions
let outdir = match env ::var_os ( " CARGO_TARGET_DIR " ) {
None = > return ,
Some ( outdir ) = > outdir ,
} ;
fs ::create_dir_all ( & outdir ) . unwrap ( ) ;
for shell in & [ Shell ::Bash , Shell ::Fish , Shell ::Zsh , Shell ::PowerShell ,
Shell ::Elvish ] {
let path = clap_complete ::generate_to ( * shell , & mut sq , " sq " , & outdir ) . unwrap ( ) ;
println! ( " cargo:warning=completion file is generated: {:?} " , path ) ;
} ;
2023-04-08 09:49:58 +02:00
build_man_pages ( ) . unwrap ( ) ;
2023-02-21 12:43:43 +01:00
}
fn dump_help ( mut cmd : clap ::Command ) -> Result < ( ) > {
cmd = cmd . term_width ( 80 ) ;
cmd . build ( ) ;
let path = PathBuf ::from ( env ::var_os ( " OUT_DIR " ) . unwrap ( ) )
. join ( " sq-usage.md " ) ;
let mut sink = fs ::File ::create ( & path )
. with_context ( | | format! ( " trying to create {} " , path . display ( ) ) ) ? ;
dump_help_inner ( & mut sink , & mut cmd , " ## " )
}
fn dump_help_inner (
sink : & mut dyn Write ,
cmd : & mut clap ::Command ,
heading : & str ,
) -> Result < ( ) > {
writeln! ( sink ) ? ;
let mut buffer = Vec ::new ( ) ;
let _ = cmd . write_long_help ( & mut buffer ) ;
let help = std ::str ::from_utf8 ( buffer . as_slice ( ) ) ? ;
2023-03-06 18:05:21 +01:00
let mut verbatim = false ;
2023-02-21 12:43:43 +01:00
for line in help . trim_end ( ) . split ( '\n' ) . skip ( 1 ) {
2023-03-06 18:05:21 +01:00
if ! verbatim & & line . starts_with ( " USAGE: " ) {
writeln! ( sink , " ```text " ) ? ;
verbatim = true ;
}
2023-02-21 12:43:43 +01:00
if line . is_empty ( ) {
writeln! ( sink ) ? ;
} else {
writeln! ( sink , " {} " , line . trim_end ( ) ) ? ;
}
}
2023-03-06 18:05:21 +01:00
if verbatim {
writeln! ( sink , " ``` " ) ? ;
}
2023-02-21 12:43:43 +01:00
// Recurse.
for subcommand in cmd
. get_subcommands_mut ( )
. filter ( | sc | sc . get_name ( ) ! = " help " )
{
writeln! ( sink ) ? ;
let heading_name = subcommand
// cmd.build() in dump_help makes sure every subcommand has a display_name
. get_display_name ( )
. unwrap ( )
. replace ( '-' , " " ) ;
writeln! ( sink , " {} Subcommand {} " , heading , heading_name ) ? ;
dump_help_inner ( sink , subcommand , & format! ( " {} # " , heading ) ) ? ;
}
Ok ( ( ) )
}
2023-04-08 09:49:58 +02:00
fn build_man_pages ( ) -> Result < ( ) > {
// Man page support.
let out_dir = std ::path ::PathBuf ::from (
std ::env ::var_os ( " OUT_DIR " )
. ok_or ( std ::io ::Error ::from ( std ::io ::ErrorKind ::NotFound ) ) ? ) ;
2023-10-20 13:04:00 +02:00
let man = clap_mangen ::Man ::new ( cli ::build ( ) ) ;
2023-04-08 09:49:58 +02:00
let mut buffer : Vec < u8 > = Default ::default ( ) ;
man . render ( & mut buffer ) ? ;
let filename = out_dir . join ( " sq.1 " ) ;
2023-06-06 15:44:32 +02:00
// println!("cargo:warning=writing man page to {}", filename.display());
2023-04-08 09:49:58 +02:00
std ::fs ::write ( filename , buffer ) ? ;
fn doit ( out_dir : & Path , prefix : & str , command : & clap ::Command ) -> Result < ( ) > {
let man = clap_mangen ::Man ::new ( command . clone ( ) ) ;
let mut buffer : Vec < u8 > = Default ::default ( ) ;
man . render ( & mut buffer ) ? ;
let filename = out_dir . join ( format! ( " {} - {} .1 " , prefix , command . get_name ( ) ) ) ;
2023-06-06 15:44:32 +02:00
// println!("cargo:warning=writing man page to {}", filename.display());
2023-04-08 09:49:58 +02:00
std ::fs ::write ( filename , buffer ) ? ;
for sc in command . get_subcommands ( ) {
doit ( out_dir ,
& format! ( " {} - {} " , prefix , command . get_name ( ) ) ,
sc ) ? ;
}
Ok ( ( ) )
}
2023-10-20 13:04:00 +02:00
for sc in cli ::build ( ) . get_subcommands ( ) {
2023-04-08 09:49:58 +02:00
doit ( & out_dir , " sq " , sc ) ? ;
}
Ok ( ( ) )
}