123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361 |
- // Copyright 2019-2021 Tauri Programme within The Commons Conservancy
- // SPDX-License-Identifier: Apache-2.0
- // SPDX-License-Identifier: MIT
- use crate::config::Config;
- use json_patch::merge;
- use serde::de::DeserializeOwned;
- use serde_json::Value;
- use std::ffi::OsStr;
- use std::path::{Path, PathBuf};
- use thiserror::Error;
- /// All extensions that are possibly supported, but perhaps not enabled.
- pub const EXTENSIONS_SUPPORTED: &[&str] = &["json", "json5", "toml"];
- /// All configuration formats that are possibly supported, but perhaps not enabled.
- pub const SUPPORTED_FORMATS: &[ConfigFormat] =
- &[ConfigFormat::Json, ConfigFormat::Json5, ConfigFormat::Toml];
- /// All configuration formats that are currently enabled.
- pub const ENABLED_FORMATS: &[ConfigFormat] = &[
- ConfigFormat::Json,
- #[cfg(feature = "config-json5")]
- ConfigFormat::Json5,
- #[cfg(feature = "config-toml")]
- ConfigFormat::Toml,
- ];
- /// The available configuration formats.
- #[derive(Debug, Copy, Clone)]
- pub enum ConfigFormat {
- /// The default JSON (tauri.conf.json) format.
- Json,
- /// The JSON5 (tauri.conf.json5) format.
- Json5,
- /// The TOML (Tauri.toml file) format.
- Toml,
- }
- impl ConfigFormat {
- /// Maps the config format to its file name.
- pub fn into_file_name(self) -> &'static str {
- match self {
- Self::Json => "tauri.conf.json",
- Self::Json5 => "tauri.conf.json5",
- Self::Toml => "Tauri.toml",
- }
- }
- fn into_platform_file_name(self) -> &'static str {
- match self {
- Self::Json => {
- if cfg!(target_os = "macos") {
- "tauri.macos.conf.json"
- } else if cfg!(windows) {
- "tauri.windows.conf.json"
- } else {
- "tauri.linux.conf.json"
- }
- }
- Self::Json5 => {
- if cfg!(target_os = "macos") {
- "tauri.macos.conf.json5"
- } else if cfg!(windows) {
- "tauri.windows.conf.json5"
- } else {
- "tauri.linux.conf.json5"
- }
- }
- Self::Toml => {
- if cfg!(target_os = "macos") {
- "Tauri.macos.toml"
- } else if cfg!(windows) {
- "Tauri.windows.toml"
- } else {
- "Tauri.linux.toml"
- }
- }
- }
- }
- }
- /// Represents all the errors that can happen while reading the config.
- #[derive(Debug, Error)]
- #[non_exhaustive]
- pub enum ConfigError {
- /// Failed to parse from JSON.
- #[error("unable to parse JSON Tauri config file at {path} because {error}")]
- FormatJson {
- /// The path that failed to parse into JSON.
- path: PathBuf,
- /// The parsing [`serde_json::Error`].
- error: serde_json::Error,
- },
- /// Failed to parse from JSON5.
- #[cfg(feature = "config-json5")]
- #[error("unable to parse JSON5 Tauri config file at {path} because {error}")]
- FormatJson5 {
- /// The path that failed to parse into JSON5.
- path: PathBuf,
- /// The parsing [`json5::Error`].
- error: ::json5::Error,
- },
- /// Failed to parse from TOML.
- #[cfg(feature = "config-toml")]
- #[error("unable to parse toml Tauri config file at {path} because {error}")]
- FormatToml {
- /// The path that failed to parse into TOML.
- path: PathBuf,
- /// The parsing [`toml::Error`].
- error: ::toml::de::Error,
- },
- /// Unknown config file name encountered.
- #[error("unsupported format encountered {0}")]
- UnsupportedFormat(String),
- /// Known file extension encountered, but corresponding parser is not enabled (cargo features).
- #[error("supported (but disabled) format encountered {extension} - try enabling `{feature}` ")]
- DisabledFormat {
- /// The extension encountered.
- extension: String,
- /// The cargo feature to enable it.
- feature: String,
- },
- /// A generic IO error with context of what caused it.
- #[error("unable to read Tauri config file at {path} because {error}")]
- Io {
- /// The path the IO error occurred on.
- path: PathBuf,
- /// The [`std::io::Error`].
- error: std::io::Error,
- },
- }
- /// Reads the configuration from the given root directory.
- ///
- /// It first looks for a `tauri.conf.json[5]` file on the given directory. The file must exist.
- /// Then it looks for a platform-specific configuration file:
- /// - `tauri.macos.conf.json[5]` on macOS
- /// - `tauri.linux.conf.json[5]` on Linux
- /// - `tauri.windows.conf.json[5]` on Windows
- /// Merging the configurations using [JSON Merge Patch (RFC 7396)].
- ///
- /// [JSON Merge Patch (RFC 7396)]: https://datatracker.ietf.org/doc/html/rfc7396.
- pub fn read_from(root_dir: PathBuf) -> Result<Value, ConfigError> {
- let mut config: Value = parse_value(root_dir.join("tauri.conf.json"))?.0;
- if let Some((platform_config, _)) = read_platform(root_dir)? {
- merge(&mut config, &platform_config);
- }
- Ok(config)
- }
- /// Reads the platform-specific configuration file from the given root directory if it exists.
- ///
- /// Check [`read_from`] for more information.
- pub fn read_platform(root_dir: PathBuf) -> Result<Option<(Value, PathBuf)>, ConfigError> {
- let platform_config_path = root_dir.join(ConfigFormat::Json.into_platform_file_name());
- if does_supported_file_name_exist(&platform_config_path) {
- let (platform_config, path): (Value, PathBuf) = parse_value(platform_config_path)?;
- Ok(Some((platform_config, path)))
- } else {
- Ok(None)
- }
- }
- /// Check if a supported config file exists at path.
- ///
- /// The passed path is expected to be the path to the "default" configuration format, in this case
- /// JSON with `.json`.
- pub fn does_supported_file_name_exist(path: impl Into<PathBuf>) -> bool {
- let path = path.into();
- let source_file_name = path.file_name().unwrap().to_str().unwrap();
- let lookup_platform_config = ENABLED_FORMATS
- .iter()
- .any(|format| source_file_name == format.into_platform_file_name());
- ENABLED_FORMATS.iter().any(|format| {
- path
- .with_file_name(if lookup_platform_config {
- format.into_platform_file_name()
- } else {
- format.into_file_name()
- })
- .exists()
- })
- }
- /// Parse the config from path, including alternative formats.
- ///
- /// Hierarchy:
- /// 1. Check if `tauri.conf.json` exists
- /// a. Parse it with `serde_json`
- /// b. Parse it with `json5` if `serde_json` fails
- /// c. Return original `serde_json` error if all above steps failed
- /// 2. Check if `tauri.conf.json5` exists
- /// a. Parse it with `json5`
- /// b. Return error if all above steps failed
- /// 3. Check if `Tauri.json` exists
- /// a. Parse it with `toml`
- /// b. Return error if all above steps failed
- /// 4. Return error if all above steps failed
- pub fn parse(path: impl Into<PathBuf>) -> Result<(Config, PathBuf), ConfigError> {
- do_parse(path.into())
- }
- /// See [`parse`] for specifics, returns a JSON [`Value`] instead of [`Config`].
- pub fn parse_value(path: impl Into<PathBuf>) -> Result<(Value, PathBuf), ConfigError> {
- do_parse(path.into())
- }
- fn do_parse<D: DeserializeOwned>(path: PathBuf) -> Result<(D, PathBuf), ConfigError> {
- let file_name = path
- .file_name()
- .map(OsStr::to_string_lossy)
- .unwrap_or_default();
- let lookup_platform_config = ENABLED_FORMATS
- .iter()
- .any(|format| file_name == format.into_platform_file_name());
- let json5 = path.with_file_name(if lookup_platform_config {
- ConfigFormat::Json5.into_platform_file_name()
- } else {
- ConfigFormat::Json5.into_file_name()
- });
- let toml = path.with_file_name(if lookup_platform_config {
- ConfigFormat::Toml.into_platform_file_name()
- } else {
- ConfigFormat::Toml.into_file_name()
- });
- let path_ext = path
- .extension()
- .map(OsStr::to_string_lossy)
- .unwrap_or_default();
- if path.exists() {
- let raw = read_to_string(&path)?;
- // to allow us to easily use the compile-time #[cfg], we always bind
- #[allow(clippy::let_and_return)]
- let json = do_parse_json(&raw, &path);
- // we also want to support **valid** json5 in the .json extension if the feature is enabled.
- // if the json5 is not valid the serde_json error for regular json will be returned.
- // this could be a bit confusing, so we may want to encourage users using json5 to use the
- // .json5 extension instead of .json
- #[cfg(feature = "config-json5")]
- let json = {
- match do_parse_json5(&raw, &path) {
- json5 @ Ok(_) => json5,
- // assume any errors from json5 in a .json file is because it's not json5
- Err(_) => json,
- }
- };
- json.map(|j| (j, path))
- } else if json5.exists() {
- #[cfg(feature = "config-json5")]
- {
- let raw = read_to_string(&json5)?;
- do_parse_json5(&raw, &path).map(|config| (config, json5))
- }
- #[cfg(not(feature = "config-json5"))]
- Err(ConfigError::DisabledFormat {
- extension: ".json5".into(),
- feature: "config-json5".into(),
- })
- } else if toml.exists() {
- #[cfg(feature = "config-toml")]
- {
- let raw = read_to_string(&toml)?;
- do_parse_toml(&raw, &path).map(|config| (config, toml))
- }
- #[cfg(not(feature = "config-toml"))]
- Err(ConfigError::DisabledFormat {
- extension: ".toml".into(),
- feature: "config-toml".into(),
- })
- } else if !EXTENSIONS_SUPPORTED.contains(&path_ext.as_ref()) {
- Err(ConfigError::UnsupportedFormat(path_ext.to_string()))
- } else {
- Err(ConfigError::Io {
- path,
- error: std::io::ErrorKind::NotFound.into(),
- })
- }
- }
- /// "Low-level" helper to parse JSON into a [`Config`].
- ///
- /// `raw` should be the contents of the file that is represented by `path`.
- pub fn parse_json(raw: &str, path: &Path) -> Result<Config, ConfigError> {
- do_parse_json(raw, path)
- }
- /// "Low-level" helper to parse JSON into a JSON [`Value`].
- ///
- /// `raw` should be the contents of the file that is represented by `path`.
- pub fn parse_json_value(raw: &str, path: &Path) -> Result<Value, ConfigError> {
- do_parse_json(raw, path)
- }
- fn do_parse_json<D: DeserializeOwned>(raw: &str, path: &Path) -> Result<D, ConfigError> {
- serde_json::from_str(raw).map_err(|error| ConfigError::FormatJson {
- path: path.into(),
- error,
- })
- }
- /// "Low-level" helper to parse JSON5 into a [`Config`].
- ///
- /// `raw` should be the contents of the file that is represented by `path`. This function requires
- /// the `config-json5` feature to be enabled.
- #[cfg(feature = "config-json5")]
- pub fn parse_json5(raw: &str, path: &Path) -> Result<Config, ConfigError> {
- do_parse_json5(raw, path)
- }
- /// "Low-level" helper to parse JSON5 into a JSON [`Value`].
- ///
- /// `raw` should be the contents of the file that is represented by `path`. This function requires
- /// the `config-json5` feature to be enabled.
- #[cfg(feature = "config-json5")]
- pub fn parse_json5_value(raw: &str, path: &Path) -> Result<Value, ConfigError> {
- do_parse_json5(raw, path)
- }
- #[cfg(feature = "config-json5")]
- fn do_parse_json5<D: DeserializeOwned>(raw: &str, path: &Path) -> Result<D, ConfigError> {
- ::json5::from_str(raw).map_err(|error| ConfigError::FormatJson5 {
- path: path.into(),
- error,
- })
- }
- #[cfg(feature = "config-toml")]
- fn do_parse_toml<D: DeserializeOwned>(raw: &str, path: &Path) -> Result<D, ConfigError> {
- ::toml::from_str(raw).map_err(|error| ConfigError::FormatToml {
- path: path.into(),
- error,
- })
- }
- /// Helper function to wrap IO errors from [`std::fs::read_to_string`] into a [`ConfigError`].
- fn read_to_string(path: &Path) -> Result<String, ConfigError> {
- std::fs::read_to_string(path).map_err(|error| ConfigError::Io {
- path: path.into(),
- error,
- })
- }
|