mirror of
https://github.com/typst/typst
synced 2025-05-14 17:15:28 +08:00
239 lines
6.9 KiB
Rust
239 lines
6.9 KiB
Rust
use std::fmt::{self, Display, Formatter};
|
|
use std::path::PathBuf;
|
|
|
|
use clap::builder::ValueParser;
|
|
use clap::{ArgAction, Args, ColorChoice, Parser, Subcommand, ValueEnum};
|
|
use semver::Version;
|
|
|
|
/// The character typically used to separate path components
|
|
/// in environment variables.
|
|
const ENV_PATH_SEP: char = if cfg!(windows) { ';' } else { ':' };
|
|
|
|
/// The Typst compiler.
|
|
#[derive(Debug, Clone, Parser)]
|
|
#[clap(name = "typst", version = crate::typst_version(), author)]
|
|
pub struct CliArguments {
|
|
/// The command to run
|
|
#[command(subcommand)]
|
|
pub command: Command,
|
|
|
|
/// Set when to use color.
|
|
/// auto = use color if a capable terminal is detected
|
|
#[clap(
|
|
long,
|
|
value_name = "WHEN",
|
|
require_equals = true,
|
|
num_args = 0..=1,
|
|
default_value = "auto",
|
|
default_missing_value = "always",
|
|
)]
|
|
pub color: ColorChoice,
|
|
|
|
/// Path to a custom CA certificate to use when making network requests.
|
|
#[clap(long = "cert", env = "TYPST_CERT")]
|
|
pub cert: Option<PathBuf>,
|
|
}
|
|
|
|
/// What to do.
|
|
#[derive(Debug, Clone, Subcommand)]
|
|
#[command()]
|
|
pub enum Command {
|
|
/// Compiles an input file into a supported output format
|
|
#[command(visible_alias = "c")]
|
|
Compile(CompileCommand),
|
|
|
|
/// Watches an input file and recompiles on changes
|
|
#[command(visible_alias = "w")]
|
|
Watch(CompileCommand),
|
|
|
|
/// Processes an input file to extract provided metadata
|
|
Query(QueryCommand),
|
|
|
|
/// Lists all discovered fonts in system and custom font paths
|
|
Fonts(FontsCommand),
|
|
|
|
/// Self update the Typst CLI
|
|
#[cfg_attr(not(feature = "self-update"), doc = " (disabled)")]
|
|
Update(UpdateCommand),
|
|
}
|
|
|
|
/// Compiles an input file into a supported output format
|
|
#[derive(Debug, Clone, Parser)]
|
|
pub struct CompileCommand {
|
|
/// Shared arguments
|
|
#[clap(flatten)]
|
|
pub common: SharedArgs,
|
|
|
|
/// Path to output file (PDF, PNG, or SVG)
|
|
pub output: Option<PathBuf>,
|
|
|
|
/// The format of the output file, inferred from the extension by default
|
|
#[arg(long = "format", short = 'f')]
|
|
pub format: Option<OutputFormat>,
|
|
|
|
/// Opens the output file using the default viewer after compilation
|
|
#[arg(long = "open")]
|
|
pub open: Option<Option<String>>,
|
|
|
|
/// The PPI (pixels per inch) to use for PNG export
|
|
#[arg(long = "ppi", default_value_t = 144.0)]
|
|
pub ppi: f32,
|
|
|
|
/// Produces performance timings of the compilation process (experimental)
|
|
///
|
|
/// The resulting JSON file can be loaded into a tracing tool such as
|
|
/// https://ui.perfetto.dev. It does not contain any sensitive information
|
|
/// apart from file names and line numbers.
|
|
#[arg(long = "timings", value_name = "OUTPUT_JSON")]
|
|
pub timings: Option<Option<PathBuf>>,
|
|
}
|
|
|
|
/// Processes an input file to extract provided metadata
|
|
#[derive(Debug, Clone, Parser)]
|
|
pub struct QueryCommand {
|
|
/// Shared arguments
|
|
#[clap(flatten)]
|
|
pub common: SharedArgs,
|
|
|
|
/// Defines which elements to retrieve
|
|
pub selector: String,
|
|
|
|
/// Extracts just one field from all retrieved elements
|
|
#[clap(long = "field")]
|
|
pub field: Option<String>,
|
|
|
|
/// Expects and retrieves exactly one element
|
|
#[clap(long = "one", default_value = "false")]
|
|
pub one: bool,
|
|
|
|
/// The format to serialize in
|
|
#[clap(long = "format", default_value = "json")]
|
|
pub format: SerializationFormat,
|
|
}
|
|
|
|
// Output file format for query command
|
|
#[derive(Debug, Copy, Clone, Eq, PartialEq, ValueEnum)]
|
|
pub enum SerializationFormat {
|
|
Json,
|
|
Yaml,
|
|
}
|
|
|
|
/// Common arguments of compile, watch, and query.
|
|
#[derive(Debug, Clone, Args)]
|
|
pub struct SharedArgs {
|
|
/// Path to input Typst file
|
|
pub input: PathBuf,
|
|
|
|
/// Configures the project root (for absolute paths)
|
|
#[clap(long = "root", env = "TYPST_ROOT", value_name = "DIR")]
|
|
pub root: Option<PathBuf>,
|
|
|
|
/// Add a string key-value pair visible through `sys.inputs`
|
|
#[clap(
|
|
long = "input",
|
|
value_name = "key=value",
|
|
action = ArgAction::Append,
|
|
value_parser = ValueParser::new(parse_input_pair),
|
|
)]
|
|
pub inputs: Vec<(String, String)>,
|
|
|
|
/// Adds additional directories to search for fonts
|
|
#[clap(
|
|
long = "font-path",
|
|
env = "TYPST_FONT_PATHS",
|
|
value_name = "DIR",
|
|
value_delimiter = ENV_PATH_SEP,
|
|
)]
|
|
pub font_paths: Vec<PathBuf>,
|
|
|
|
/// The format to emit diagnostics in
|
|
#[clap(
|
|
long,
|
|
default_value_t = DiagnosticFormat::Human,
|
|
value_parser = clap::value_parser!(DiagnosticFormat)
|
|
)]
|
|
pub diagnostic_format: DiagnosticFormat,
|
|
}
|
|
|
|
/// Parses key/value pairs split by the first equal sign.
|
|
///
|
|
/// This function will return an error if the argument contains no equals sign
|
|
/// or contains the key (before the equals sign) is empty.
|
|
fn parse_input_pair(raw: &str) -> Result<(String, String), String> {
|
|
let (key, val) = raw
|
|
.split_once('=')
|
|
.ok_or("input must be a key and a value separated by an equal sign")?;
|
|
let key = key.trim().to_owned();
|
|
if key.is_empty() {
|
|
return Err("the key was missing or empty".to_owned());
|
|
}
|
|
let val = val.trim().to_owned();
|
|
Ok((key, val))
|
|
}
|
|
|
|
/// Lists all discovered fonts in system and custom font paths
|
|
#[derive(Debug, Clone, Parser)]
|
|
pub struct FontsCommand {
|
|
/// Adds additional directories to search for fonts
|
|
#[clap(
|
|
long = "font-path",
|
|
env = "TYPST_FONT_PATHS",
|
|
value_name = "DIR",
|
|
value_delimiter = ENV_PATH_SEP,
|
|
)]
|
|
pub font_paths: Vec<PathBuf>,
|
|
|
|
/// Also lists style variants of each font family
|
|
#[arg(long)]
|
|
pub variants: bool,
|
|
}
|
|
|
|
/// Which format to use for diagnostics.
|
|
#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd, ValueEnum)]
|
|
pub enum DiagnosticFormat {
|
|
Human,
|
|
Short,
|
|
}
|
|
|
|
impl Display for DiagnosticFormat {
|
|
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
|
|
self.to_possible_value()
|
|
.expect("no values are skipped")
|
|
.get_name()
|
|
.fmt(f)
|
|
}
|
|
}
|
|
|
|
/// Update the CLI using a pre-compiled binary from a Typst GitHub release.
|
|
#[derive(Debug, Clone, Parser)]
|
|
pub struct UpdateCommand {
|
|
/// Which version to update to (defaults to latest)
|
|
pub version: Option<Version>,
|
|
|
|
/// Forces a downgrade to an older version (required for downgrading)
|
|
#[clap(long, default_value_t = false)]
|
|
pub force: bool,
|
|
|
|
/// Reverts to the version from before the last update (only possible if
|
|
/// `typst update` has previously ran)
|
|
#[clap(long, default_value_t = false, exclusive = true)]
|
|
pub revert: bool,
|
|
}
|
|
|
|
/// Which format to use for the generated output file.
|
|
#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd, ValueEnum)]
|
|
pub enum OutputFormat {
|
|
Pdf,
|
|
Png,
|
|
Svg,
|
|
}
|
|
|
|
impl Display for OutputFormat {
|
|
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
|
|
self.to_possible_value()
|
|
.expect("no values are skipped")
|
|
.get_name()
|
|
.fmt(f)
|
|
}
|
|
}
|