2024-03-01 11:29:01 +00:00
|
|
|
// Copyright 2019-2024 Tauri Programme within The Commons Conservancy
|
2021-12-09 15:21:33 +00:00
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
|
|
2024-08-27 15:05:19 +00:00
|
|
|
use super::PluginIosFramework;
|
2024-03-19 13:41:41 +00:00
|
|
|
use crate::helpers::prompts;
|
2021-12-09 15:21:33 +00:00
|
|
|
use crate::Result;
|
|
|
|
|
use crate::{
|
2022-05-07 13:19:54 +00:00
|
|
|
helpers::{resolve_tauri_path, template},
|
2021-12-09 15:21:33 +00:00
|
|
|
VersionMetadata,
|
|
|
|
|
};
|
|
|
|
|
use anyhow::Context;
|
2024-08-27 15:05:19 +00:00
|
|
|
use clap::Parser;
|
2021-12-09 15:21:33 +00:00
|
|
|
use handlebars::{to_json, Handlebars};
|
2023-10-11 15:34:33 +00:00
|
|
|
use heck::{ToKebabCase, ToPascalCase, ToSnakeCase};
|
2021-12-09 15:21:33 +00:00
|
|
|
use include_dir::{include_dir, Dir};
|
2024-06-26 14:11:31 +00:00
|
|
|
use std::ffi::{OsStr, OsString};
|
2023-02-06 11:56:00 +00:00
|
|
|
use std::{
|
2023-02-13 13:54:48 +00:00
|
|
|
collections::BTreeMap,
|
|
|
|
|
env::current_dir,
|
|
|
|
|
fs::{create_dir_all, remove_dir_all, File, OpenOptions},
|
2023-02-16 15:24:49 +00:00
|
|
|
path::{Component, Path, PathBuf},
|
2023-02-06 11:56:00 +00:00
|
|
|
};
|
2021-12-09 15:21:33 +00:00
|
|
|
|
2024-04-01 14:43:52 +00:00
|
|
|
pub const TEMPLATE_DIR: Dir<'_> = include_dir!("$CARGO_MANIFEST_DIR/templates/plugin");
|
2021-12-09 15:21:33 +00:00
|
|
|
|
|
|
|
|
#[derive(Debug, Parser)]
|
2023-10-16 12:44:23 +00:00
|
|
|
#[clap(about = "Initialize a Tauri plugin project on an existing directory")]
|
2021-12-09 15:21:33 +00:00
|
|
|
pub struct Options {
|
2023-10-11 15:34:33 +00:00
|
|
|
/// Name of your Tauri plugin.
|
2024-03-11 14:25:20 +00:00
|
|
|
/// If not specified, it will be inferred from the current directory.
|
2023-10-11 15:34:33 +00:00
|
|
|
pub(crate) plugin_name: Option<String>,
|
2023-02-16 15:24:49 +00:00
|
|
|
/// Initializes a Tauri plugin without the TypeScript API
|
2022-01-03 23:00:58 +00:00
|
|
|
#[clap(long)]
|
2023-10-11 15:34:33 +00:00
|
|
|
pub(crate) no_api: bool,
|
2021-12-09 15:21:33 +00:00
|
|
|
/// Initializes a Tauri core plugin (internal usage)
|
2022-02-08 16:13:46 +00:00
|
|
|
#[clap(long, hide(true))]
|
2023-10-11 15:34:33 +00:00
|
|
|
pub(crate) tauri: bool,
|
2021-12-09 15:21:33 +00:00
|
|
|
/// Set target directory for init
|
|
|
|
|
#[clap(short, long)]
|
|
|
|
|
#[clap(default_value_t = current_dir().expect("failed to read cwd").display().to_string())]
|
2023-10-11 15:34:33 +00:00
|
|
|
pub(crate) directory: String,
|
2021-12-09 15:21:33 +00:00
|
|
|
/// Path of the Tauri project to use (relative to the cwd)
|
|
|
|
|
#[clap(short, long)]
|
2023-10-11 15:34:33 +00:00
|
|
|
pub(crate) tauri_path: Option<PathBuf>,
|
2021-12-09 15:21:33 +00:00
|
|
|
/// Author name
|
|
|
|
|
#[clap(short, long)]
|
2023-10-11 15:34:33 +00:00
|
|
|
pub(crate) author: Option<String>,
|
|
|
|
|
/// Whether to initialize an Android project for the plugin.
|
|
|
|
|
#[clap(long)]
|
|
|
|
|
pub(crate) android: bool,
|
|
|
|
|
/// Whether to initialize an iOS project for the plugin.
|
|
|
|
|
#[clap(long)]
|
|
|
|
|
pub(crate) ios: bool,
|
|
|
|
|
/// Whether to initialize Android and iOS projects for the plugin.
|
|
|
|
|
#[clap(long)]
|
|
|
|
|
pub(crate) mobile: bool,
|
2024-05-24 11:23:41 +00:00
|
|
|
/// Type of framework to use for the iOS project.
|
|
|
|
|
#[clap(long)]
|
2024-08-27 15:05:19 +00:00
|
|
|
#[clap(default_value_t = PluginIosFramework::default())]
|
|
|
|
|
pub(crate) ios_framework: PluginIosFramework,
|
2021-12-09 15:21:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Options {
|
|
|
|
|
fn load(&mut self) {
|
|
|
|
|
if self.author.is_none() {
|
|
|
|
|
self.author.replace(if self.tauri {
|
|
|
|
|
"Tauri Programme within The Commons Conservancy".into()
|
|
|
|
|
} else {
|
|
|
|
|
"You".into()
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn command(mut options: Options) -> Result<()> {
|
|
|
|
|
options.load();
|
2023-10-11 15:34:33 +00:00
|
|
|
|
|
|
|
|
let plugin_name = match options.plugin_name {
|
|
|
|
|
None => super::infer_plugin_name(&options.directory)?,
|
|
|
|
|
Some(name) => name,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let template_target_path = PathBuf::from(options.directory);
|
2023-02-16 15:24:49 +00:00
|
|
|
let metadata = crates_metadata()?;
|
2023-10-11 15:34:33 +00:00
|
|
|
if std::fs::read_dir(&template_target_path)?.count() > 0 {
|
2024-03-04 19:22:07 +00:00
|
|
|
log::warn!("Plugin dir ({:?}) not empty.", template_target_path);
|
2021-12-09 15:21:33 +00:00
|
|
|
} else {
|
2024-02-01 11:53:56 +00:00
|
|
|
let (tauri_dep, tauri_example_dep, tauri_build_dep, tauri_plugin_dep) =
|
2021-12-09 15:21:33 +00:00
|
|
|
if let Some(tauri_path) = options.tauri_path {
|
|
|
|
|
(
|
|
|
|
|
format!(
|
|
|
|
|
r#"{{ path = {:?} }}"#,
|
|
|
|
|
resolve_tauri_path(&tauri_path, "core/tauri")
|
|
|
|
|
),
|
|
|
|
|
format!(
|
2022-11-03 21:57:32 +00:00
|
|
|
r#"{{ path = {:?} }}"#,
|
2021-12-09 15:21:33 +00:00
|
|
|
resolve_tauri_path(&tauri_path, "core/tauri")
|
|
|
|
|
),
|
|
|
|
|
format!(
|
2024-02-02 17:11:59 +00:00
|
|
|
"{{ path = {:?}, default-features = false }}",
|
2021-12-09 15:21:33 +00:00
|
|
|
resolve_tauri_path(&tauri_path, "core/tauri-build")
|
|
|
|
|
),
|
2024-02-01 11:53:56 +00:00
|
|
|
format!(
|
|
|
|
|
r#"{{ path = {:?}, features = ["build"] }}"#,
|
|
|
|
|
resolve_tauri_path(&tauri_path, "core/tauri-plugin")
|
|
|
|
|
),
|
2021-12-09 15:21:33 +00:00
|
|
|
)
|
|
|
|
|
} else {
|
|
|
|
|
(
|
|
|
|
|
format!(r#"{{ version = "{}" }}"#, metadata.tauri),
|
2022-11-03 21:57:32 +00:00
|
|
|
format!(r#"{{ version = "{}" }}"#, metadata.tauri),
|
2024-02-02 17:11:59 +00:00
|
|
|
format!(
|
|
|
|
|
r#"{{ version = "{}", default-features = false }}"#,
|
|
|
|
|
metadata.tauri_build
|
|
|
|
|
),
|
2024-02-01 11:53:56 +00:00
|
|
|
format!(
|
|
|
|
|
r#"{{ version = "{}", features = ["build"] }}"#,
|
|
|
|
|
metadata.tauri_plugin
|
|
|
|
|
),
|
2021-12-09 15:21:33 +00:00
|
|
|
)
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let _ = remove_dir_all(&template_target_path);
|
2023-05-02 12:59:59 +00:00
|
|
|
let mut handlebars = Handlebars::new();
|
Merge remote-tracking branch 'origin/dev' into next (#7067)
Co-authored-by: wusyong <wusyong@users.noreply.github.com>
Co-authored-by: Fabian-Lars <fabianlars@fabianlars.de>
Co-authored-by: Lucas Nogueira <lucas@tauri.studio>
Co-authored-by: Simon Hyll <hyllsimon@gmail.com>
Co-authored-by: Lucas Fernandes Nogueira <lucas@tauri.app>
Co-authored-by: Lucas Nogueira <lucas@tauri.app>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com>
Co-authored-by: Lucas Fernandes Nogueira <lucas@tauri.studio>
Co-authored-by: Amr Bashir <amr.bashir2015@gmail.com>
Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com>
Co-authored-by: chip <chip@chip.sh>
Co-authored-by: Raphii <iam@raphii.co>
Co-authored-by: Ronie Martinez <ronmarti18@gmail.com>
Co-authored-by: hanaTsuk1 <101488209+hanaTsuk1@users.noreply.github.com>
Co-authored-by: nathan-fall <39990940+nathan-fall@users.noreply.github.com>
Co-authored-by: Akshay <nerdy@peppe.rs>
Co-authored-by: KurikoMoe <kurikomoe@gmail.com>
Co-authored-by: Guilherme Oenning <me@goenning.net>
Co-authored-by: Pierre Cashon <biaocy91@gmail.com>
Co-authored-by: Jack Wills <32690432+mrjackwills@users.noreply.github.com>
Co-authored-by: Amirhossein Akhlaghpour <m9.akhlaghpoor@gmail.com>
Co-authored-by: Risto Stevcev <me@risto.codes>
Co-authored-by: Soumt <rltks1305@naver.com>
Co-authored-by: yutotnh <57719497+yutotnh@users.noreply.github.com>
Co-authored-by: Gökçe Merdun <agmmnn@gmail.com>
Co-authored-by: Nathanael Rea <Nathan@NathanaelRea.com>
Co-authored-by: Usman Rajab <usman.rajab@gmail.com>
Co-authored-by: Francis The Basilisk <36006338+snorkysnark@users.noreply.github.com>
Co-authored-by: Lej77 <31554212+Lej77@users.noreply.github.com>
Co-authored-by: Tomáš Diblík <dibla.tomas@post.cz>
Co-authored-by: Jonas Kruckenberg <iterpre@protonmail.com>
Co-authored-by: Pascal Sommer <Pascal-So@users.noreply.github.com>
Co-authored-by: Bo <bertonzh@gmail.com>
Co-authored-by: Kevin Yue <k3vinyue@gmail.com>
fixed grammar and typos (#6937)
Fix api.js docs pipeline with updated typedoc dependencies (#6945)
closes #6887 (#6922)
fix(core): Fix `WindowBuilder::on_navigation` handler never registerd, closes #6865 (#6921)
fix(core): Fix `WindowBuilder::on_navigation` handler never registerd, closes #6865
fix broken symlinks in license files (#6336)
fix(cli): fix cli connection timeout to dev server (fix #6045) (#6046)
fix(bundler): ensure that there are no duplicate extension arguments when bundling on Windows, fixes #6103 (#6917)
fix(bundler): ensure that there are no duplicate extension arguments during bundling on Windows (fix #6103)
closes #5491 (#6408)
fix(nsis): prefill $INSTDIR with previous install path and respect `/D` flag, closes #6928 (#6935)
fix(nsis): prefill $INSTDIR with previous install path and respect `/D` flag, closes #6928
fix(updater): emit `UPTODATE` when server responds with 204, closes #6934 (#6970)
fix(core): unpin all dependencies, closes #6944 (#6966)
fix(bundler): Add new lang_file option in persian variant. (#6972)
fix(core/ipc): access url through webview native object, closes #6889 (#6976)
fix(core): remove trailing slash in http scope url, closes #5208 (#6974)
fix(core): remove trailing slash in http scope url, closes #5208
fix(cli): find correct binary when `--profile` is used, closes #6954 (#6979)
fix(cli): find correct binary when `--profile` is used, closes #6954
closes #6955 (#6987)
closes #6955
closes #6158 (#6969)
closes #6158
fix(cli): improve vs build tools detection (#6982)
fix: updated appimage script to follow symlinks for /usr/lib* (fix: #6992) (#6996)
fix(cli): correctly remove Cargo features (#7013)
Fix typo (#7012)
fix(cli): revert metadata.json field rename from #6795 (#7029)
closes #6732 (#6736)
fix: add missing file properties on Windows, closes #6676 (#6693)
fix(cli.js): detect node-20 binary (#6667)
fix version-or-publish workflow (#7031)
fix(cli/devserver): inject autoreload into HTML only, closes #6997 (#7032)
fix(bundler/nsis): write installer templates UTF16LE encoded, closes #7036 (#7040)
fix(bundler/nsis): write installer templates UTF16LE encoded, closes #7036
fix(core): rewrite `asset` protocol streaming, closes #6375 (#6390)
closes #5939 (#5960)
fix(core): use `safe_block_on` (#7047)
closes #6859 (#6933)
closes #6955 (#6998)
fix(core): populate webview_attrs from config, closes #6794 (#6797)
closes #5176 (#5180)
fix: sound for notifications on windows (fix #6652) (#6680)
close native window's buttons, closes #2353 (#6665)
fix(bundler/nsis): calculate accurate app size, closes #7056 (#7057)
fix(tests): only download update when it is available (#7061)
closes #6706 (#6712)
fix(doc): correct the doc of `content_protected()` (#7065)
closes #6472 (#6530)
fix(macros): use full path to Result to avoid issues with type aliases (#7071)
2023-05-30 00:29:24 +00:00
|
|
|
handlebars.register_escape_fn(handlebars::no_escape);
|
2021-12-09 15:21:33 +00:00
|
|
|
|
|
|
|
|
let mut data = BTreeMap::new();
|
2023-10-11 15:34:33 +00:00
|
|
|
plugin_name_data(&mut data, &plugin_name);
|
2021-12-09 15:21:33 +00:00
|
|
|
data.insert("tauri_dep", to_json(tauri_dep));
|
|
|
|
|
data.insert("tauri_example_dep", to_json(tauri_example_dep));
|
|
|
|
|
data.insert("tauri_build_dep", to_json(tauri_build_dep));
|
2024-02-01 11:53:56 +00:00
|
|
|
data.insert("tauri_plugin_dep", to_json(tauri_plugin_dep));
|
2021-12-09 15:21:33 +00:00
|
|
|
data.insert("author", to_json(options.author));
|
|
|
|
|
|
|
|
|
|
if options.tauri {
|
|
|
|
|
data.insert(
|
|
|
|
|
"license_header",
|
|
|
|
|
to_json(
|
2024-03-01 11:29:01 +00:00
|
|
|
"// Copyright 2019-2024 Tauri Programme within The Commons Conservancy
|
2021-12-09 15:21:33 +00:00
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
// SPDX-License-Identifier: MIT\n\n"
|
|
|
|
|
.replace(" ", "")
|
|
|
|
|
.replace(" //", "//"),
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-10-11 15:34:33 +00:00
|
|
|
let plugin_id = if options.android || options.mobile {
|
2024-03-19 13:41:41 +00:00
|
|
|
let plugin_id = prompts::input(
|
2023-10-11 15:34:33 +00:00
|
|
|
"What should be the Android Package ID for your plugin?",
|
|
|
|
|
Some(format!("com.plugin.{}", plugin_name)),
|
|
|
|
|
false,
|
|
|
|
|
false,
|
|
|
|
|
)?
|
|
|
|
|
.unwrap();
|
2023-02-16 15:24:49 +00:00
|
|
|
|
2023-10-11 15:34:33 +00:00
|
|
|
data.insert("android_package_id", to_json(&plugin_id));
|
|
|
|
|
Some(plugin_id)
|
|
|
|
|
} else {
|
|
|
|
|
None
|
|
|
|
|
};
|
2023-02-16 15:24:49 +00:00
|
|
|
|
2024-08-27 15:05:19 +00:00
|
|
|
let ios_framework = options.ios_framework;
|
2024-05-24 11:23:41 +00:00
|
|
|
|
2023-02-16 15:24:49 +00:00
|
|
|
let mut created_dirs = Vec::new();
|
|
|
|
|
template::render_with_generator(
|
2021-12-09 15:21:33 +00:00
|
|
|
&handlebars,
|
|
|
|
|
&data,
|
2023-02-16 15:24:49 +00:00
|
|
|
&TEMPLATE_DIR,
|
2021-12-09 15:21:33 +00:00
|
|
|
&template_target_path,
|
2023-02-16 15:24:49 +00:00
|
|
|
&mut |mut path| {
|
|
|
|
|
let mut components = path.components();
|
|
|
|
|
let root = components.next().unwrap();
|
2023-02-11 13:30:44 +00:00
|
|
|
|
2023-02-16 15:24:49 +00:00
|
|
|
if let Component::Normal(component) = root {
|
|
|
|
|
match component.to_str().unwrap() {
|
|
|
|
|
"__example-api" => {
|
|
|
|
|
if options.no_api {
|
|
|
|
|
return Ok(None);
|
|
|
|
|
} else {
|
|
|
|
|
path = Path::new("examples").join(components.collect::<PathBuf>());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
"__example-basic" => {
|
|
|
|
|
if options.no_api {
|
|
|
|
|
path = Path::new("examples").join(components.collect::<PathBuf>());
|
|
|
|
|
} else {
|
|
|
|
|
return Ok(None);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
"android" => {
|
2023-10-11 15:34:33 +00:00
|
|
|
if options.android || options.mobile {
|
|
|
|
|
return generate_android_out_file(
|
|
|
|
|
&path,
|
|
|
|
|
&template_target_path,
|
|
|
|
|
&plugin_id.as_ref().unwrap().replace('.', "/"),
|
|
|
|
|
&mut created_dirs,
|
|
|
|
|
);
|
|
|
|
|
} else {
|
|
|
|
|
return Ok(None);
|
|
|
|
|
}
|
2023-02-16 15:24:49 +00:00
|
|
|
}
|
2024-05-24 11:23:41 +00:00
|
|
|
"ios-spm" | "ios-xcode" if !(options.ios || options.mobile) => return Ok(None),
|
2024-08-27 15:05:19 +00:00
|
|
|
"ios-spm" if !matches!(ios_framework, PluginIosFramework::Spm) => return Ok(None),
|
|
|
|
|
"ios-xcode" if !matches!(ios_framework, PluginIosFramework::Xcode) => return Ok(None),
|
2024-05-24 11:23:41 +00:00
|
|
|
"ios-spm" | "ios-xcode" => {
|
|
|
|
|
let folder_name = components.next().unwrap().as_os_str().to_string_lossy();
|
2024-06-26 14:11:31 +00:00
|
|
|
let new_folder_name = folder_name.replace("{{ plugin_name }}", &plugin_name);
|
|
|
|
|
let new_folder_name = OsString::from(&new_folder_name);
|
|
|
|
|
|
|
|
|
|
path = [
|
|
|
|
|
Component::Normal(OsStr::new("ios")),
|
|
|
|
|
Component::Normal(&new_folder_name),
|
|
|
|
|
]
|
|
|
|
|
.into_iter()
|
|
|
|
|
.chain(components)
|
|
|
|
|
.collect::<PathBuf>();
|
2024-05-24 11:23:41 +00:00
|
|
|
}
|
2024-04-02 16:59:26 +00:00
|
|
|
"guest-js" | "rollup.config.js" | "tsconfig.json" | "package.json"
|
|
|
|
|
if options.no_api =>
|
|
|
|
|
{
|
|
|
|
|
return Ok(None);
|
2023-02-16 15:24:49 +00:00
|
|
|
}
|
|
|
|
|
_ => (),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let path = template_target_path.join(path);
|
|
|
|
|
let parent = path.parent().unwrap().to_path_buf();
|
|
|
|
|
if !created_dirs.contains(&parent) {
|
|
|
|
|
create_dir_all(&parent)?;
|
|
|
|
|
created_dirs.push(parent);
|
|
|
|
|
}
|
|
|
|
|
File::create(path).map(Some)
|
|
|
|
|
},
|
|
|
|
|
)
|
2024-06-26 14:11:31 +00:00
|
|
|
.with_context(|| "failed to render plugin template")?;
|
2021-12-09 15:21:33 +00:00
|
|
|
}
|
2024-02-26 17:42:13 +00:00
|
|
|
|
2024-06-11 04:06:46 +00:00
|
|
|
let permissions_dir = template_target_path.join("permissions");
|
|
|
|
|
std::fs::create_dir(&permissions_dir)
|
2024-02-26 17:42:13 +00:00
|
|
|
.with_context(|| "failed to create `permissions` directory")?;
|
|
|
|
|
|
2024-06-11 04:06:46 +00:00
|
|
|
let default_permissions = r#"[default]
|
|
|
|
|
description = "Default permissions for the plugin"
|
|
|
|
|
permissions = ["allow-ping"]
|
|
|
|
|
"#;
|
|
|
|
|
std::fs::write(permissions_dir.join("default.toml"), default_permissions)
|
|
|
|
|
.with_context(|| "failed to write `permissions/default.toml`")?;
|
|
|
|
|
|
2021-12-09 15:21:33 +00:00
|
|
|
Ok(())
|
|
|
|
|
}
|
2023-02-06 11:56:00 +00:00
|
|
|
|
2023-02-16 15:24:49 +00:00
|
|
|
pub fn plugin_name_data(data: &mut BTreeMap<&'static str, serde_json::Value>, plugin_name: &str) {
|
|
|
|
|
data.insert("plugin_name_original", to_json(plugin_name));
|
|
|
|
|
data.insert("plugin_name", to_json(plugin_name.to_kebab_case()));
|
|
|
|
|
data.insert(
|
|
|
|
|
"plugin_name_snake_case",
|
|
|
|
|
to_json(plugin_name.to_snake_case()),
|
|
|
|
|
);
|
2023-02-16 16:33:50 +00:00
|
|
|
data.insert(
|
|
|
|
|
"plugin_name_pascal_case",
|
|
|
|
|
to_json(plugin_name.to_pascal_case()),
|
|
|
|
|
);
|
2023-02-16 15:24:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn crates_metadata() -> Result<VersionMetadata> {
|
2023-06-22 13:43:47 +00:00
|
|
|
serde_json::from_str::<VersionMetadata>(include_str!("../../metadata-v2.json"))
|
|
|
|
|
.map_err(Into::into)
|
2023-02-16 15:24:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn generate_android_out_file(
|
2023-02-13 13:54:48 +00:00
|
|
|
path: &Path,
|
|
|
|
|
dest: &Path,
|
|
|
|
|
package_path: &str,
|
|
|
|
|
created_dirs: &mut Vec<PathBuf>,
|
|
|
|
|
) -> std::io::Result<Option<File>> {
|
|
|
|
|
let mut iter = path.iter();
|
|
|
|
|
let root = iter.next().unwrap().to_str().unwrap();
|
|
|
|
|
let path = match (root, path.extension().and_then(|o| o.to_str())) {
|
|
|
|
|
("src", Some("kt")) => {
|
|
|
|
|
let parent = path.parent().unwrap();
|
|
|
|
|
let file_name = path.file_name().unwrap();
|
|
|
|
|
let out_dir = dest.join(parent).join(package_path);
|
|
|
|
|
out_dir.join(file_name)
|
|
|
|
|
}
|
|
|
|
|
_ => dest.join(path),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let parent = path.parent().unwrap().to_path_buf();
|
|
|
|
|
if !created_dirs.contains(&parent) {
|
|
|
|
|
create_dir_all(&parent)?;
|
|
|
|
|
created_dirs.push(parent);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let mut options = OpenOptions::new();
|
|
|
|
|
options.write(true);
|
|
|
|
|
|
|
|
|
|
#[cfg(unix)]
|
2024-03-04 19:22:07 +00:00
|
|
|
if path.file_name().unwrap() == std::ffi::OsStr::new("gradlew") {
|
2023-02-13 13:54:48 +00:00
|
|
|
use std::os::unix::fs::OpenOptionsExt;
|
|
|
|
|
options.mode(0o755);
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-28 16:42:21 +00:00
|
|
|
if !path.exists() {
|
2023-02-13 13:54:48 +00:00
|
|
|
options.create(true).open(path).map(Some)
|
|
|
|
|
} else {
|
|
|
|
|
Ok(None)
|
|
|
|
|
}
|
|
|
|
|
}
|