Added installer
This commit is contained in:
commit
1f40ae9376
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
||||
/target
|
61
.gitlab-ci.yml
Normal file
61
.gitlab-ci.yml
Normal file
@ -0,0 +1,61 @@
|
||||
stages:
|
||||
- build
|
||||
- release
|
||||
|
||||
variables:
|
||||
PACKAGE_REGISTRY_URL: "${CI_API_V4_URL}/projects/${CI_PROJECT_ID}/packages/generic/insatller/"
|
||||
|
||||
build_glibc:
|
||||
stage: build
|
||||
image: rust:bullseye
|
||||
script:
|
||||
- cargo build --release
|
||||
- cp target/release/dotfiles_loader ../installer-amd64-glibc
|
||||
artifacts:
|
||||
paths:
|
||||
- installer-amd64-glibc
|
||||
|
||||
build_muslc:
|
||||
stage: build
|
||||
image: rust:alpine
|
||||
script:
|
||||
- apk add pkgconf musl-dev
|
||||
- cd backend
|
||||
- cargo build --release
|
||||
- cp target/release/dotfiles_loader ../installer-amd64-muslc
|
||||
artifacts:
|
||||
paths:
|
||||
- installer-amd64-muslc
|
||||
|
||||
|
||||
upload_assets:
|
||||
stage: release
|
||||
image: curlimages/curl:latest
|
||||
needs:
|
||||
- job: build_glibc
|
||||
artifacts: true
|
||||
- job: build_muslc
|
||||
artifacts: true
|
||||
script:
|
||||
- 'curl --header "JOB-TOKEN: $CI_JOB_TOKEN" --upload-file installer-amd64-glibc "${PACKAGE_REGISTRY_URL}/dev-$CI_COMMIT_SHORT_SHA/installer-amd64-glibc"'
|
||||
- 'curl --header "JOB-TOKEN: $CI_JOB_TOKEN" --upload-file installer-amd64-muslc "${PACKAGE_REGISTRY_URL}/dev-$CI_COMMIT_SHORT_SHA/installer-amd64-muslc"'
|
||||
|
||||
create_release:
|
||||
stage: release
|
||||
image: registry.gitlab.com/gitlab-org/release-cli:latest
|
||||
needs:
|
||||
- upload_assets
|
||||
script:
|
||||
- echo "running release_job"
|
||||
release: # See https://docs.gitlab.com/ee/ci/yaml/#release for available properties
|
||||
tag_name: 'dev-$CI_COMMIT_SHORT_SHA'
|
||||
description: 'Release dev-$CI_COMMIT_SHORT_SHA'
|
||||
assets:
|
||||
links:
|
||||
- name: 'installer-amd64-glibc'
|
||||
url: '${PACKAGE_REGISTRY_URL}/dev-$CI_COMMIT_SHORT_SHA/installer-amd64-glibc'
|
||||
link_type: package
|
||||
- name: 'installer-amd64-muslc'
|
||||
url: '${PACKAGE_REGISTRY_URL}/dev-$CI_COMMIT_SHORT_SHA/installer-amd64-muslc'
|
||||
link_type: package
|
||||
|
8
.idea/.gitignore
vendored
Normal file
8
.idea/.gitignore
vendored
Normal file
@ -0,0 +1,8 @@
|
||||
# Default ignored files
|
||||
/shelf/
|
||||
/workspace.xml
|
||||
# Editor-based HTTP Client requests
|
||||
/httpRequests/
|
||||
# Datasource local storage ignored files
|
||||
/dataSources/
|
||||
/dataSources.local.xml
|
11
.idea/dotfiles_loader.iml
Normal file
11
.idea/dotfiles_loader.iml
Normal file
@ -0,0 +1,11 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<module type="CPP_MODULE" version="4">
|
||||
<component name="NewModuleRootManager">
|
||||
<content url="file://$MODULE_DIR$">
|
||||
<sourceFolder url="file://$MODULE_DIR$/src" isTestSource="false" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/target" />
|
||||
</content>
|
||||
<orderEntry type="inheritedJdk" />
|
||||
<orderEntry type="sourceFolder" forTests="false" />
|
||||
</component>
|
||||
</module>
|
8
.idea/modules.xml
Normal file
8
.idea/modules.xml
Normal file
@ -0,0 +1,8 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="ProjectModuleManager">
|
||||
<modules>
|
||||
<module fileurl="file://$PROJECT_DIR$/.idea/dotfiles_loader.iml" filepath="$PROJECT_DIR$/.idea/dotfiles_loader.iml" />
|
||||
</modules>
|
||||
</component>
|
||||
</project>
|
6
.idea/vcs.xml
Normal file
6
.idea/vcs.xml
Normal file
@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="VcsDirectoryMappings">
|
||||
<mapping directory="" vcs="Git" />
|
||||
</component>
|
||||
</project>
|
531
Cargo.lock
generated
Normal file
531
Cargo.lock
generated
Normal file
@ -0,0 +1,531 @@
|
||||
# This file is automatically @generated by Cargo.
|
||||
# It is not intended for manual editing.
|
||||
version = 3
|
||||
|
||||
[[package]]
|
||||
name = "autocfg"
|
||||
version = "1.1.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa"
|
||||
|
||||
[[package]]
|
||||
name = "bitflags"
|
||||
version = "1.3.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
|
||||
|
||||
[[package]]
|
||||
name = "cc"
|
||||
version = "1.0.78"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "a20104e2335ce8a659d6dd92a51a767a0c062599c73b343fd152cb401e828c3d"
|
||||
dependencies = [
|
||||
"jobserver",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "cfg-if"
|
||||
version = "1.0.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
|
||||
|
||||
[[package]]
|
||||
name = "console"
|
||||
version = "0.15.5"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "c3d79fbe8970a77e3e34151cc13d3b3e248aa0faaecb9f6091fa07ebefe5ad60"
|
||||
dependencies = [
|
||||
"encode_unicode",
|
||||
"lazy_static",
|
||||
"libc",
|
||||
"unicode-width",
|
||||
"windows-sys",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "dialoguer"
|
||||
version = "0.10.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "af3c796f3b0b408d9fd581611b47fa850821fcb84aa640b83a3c1a5be2d691f2"
|
||||
dependencies = [
|
||||
"console",
|
||||
"shell-words",
|
||||
"tempfile",
|
||||
"zeroize",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "dotfiles_loader"
|
||||
version = "0.1.0"
|
||||
dependencies = [
|
||||
"dialoguer",
|
||||
"fs_extra",
|
||||
"git2",
|
||||
"indicatif",
|
||||
"serde",
|
||||
"strum",
|
||||
"toml",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "encode_unicode"
|
||||
version = "0.3.6"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "a357d28ed41a50f9c765dbfe56cbc04a64e53e5fc58ba79fbc34c10ef3df831f"
|
||||
|
||||
[[package]]
|
||||
name = "fastrand"
|
||||
version = "1.8.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "a7a407cfaa3385c4ae6b23e84623d48c2798d06e3e6a1878f7f59f17b3f86499"
|
||||
dependencies = [
|
||||
"instant",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "form_urlencoded"
|
||||
version = "1.1.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "a9c384f161156f5260c24a097c56119f9be8c798586aecc13afbcbe7b7e26bf8"
|
||||
dependencies = [
|
||||
"percent-encoding",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "fs_extra"
|
||||
version = "1.2.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "2022715d62ab30faffd124d40b76f4134a550a87792276512b18d63272333394"
|
||||
|
||||
[[package]]
|
||||
name = "git2"
|
||||
version = "0.16.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "be36bc9e0546df253c0cc41fd0af34f5e92845ad8509462ec76672fac6997f5b"
|
||||
dependencies = [
|
||||
"bitflags",
|
||||
"libc",
|
||||
"libgit2-sys",
|
||||
"log",
|
||||
"openssl-probe",
|
||||
"openssl-sys",
|
||||
"url",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "heck"
|
||||
version = "0.4.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "2540771e65fc8cb83cd6e8a237f70c319bd5c29f78ed1084ba5d50eeac86f7f9"
|
||||
|
||||
[[package]]
|
||||
name = "idna"
|
||||
version = "0.3.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "e14ddfc70884202db2244c223200c204c2bda1bc6e0998d11b5e024d657209e6"
|
||||
dependencies = [
|
||||
"unicode-bidi",
|
||||
"unicode-normalization",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "indicatif"
|
||||
version = "0.17.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "cef509aa9bc73864d6756f0d34d35504af3cf0844373afe9b8669a5b8005a729"
|
||||
dependencies = [
|
||||
"console",
|
||||
"number_prefix",
|
||||
"portable-atomic",
|
||||
"unicode-width",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "instant"
|
||||
version = "0.1.12"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "7a5bbe824c507c5da5956355e86a746d82e0e1464f65d862cc5e71da70e94b2c"
|
||||
dependencies = [
|
||||
"cfg-if",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "jobserver"
|
||||
version = "0.1.25"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "068b1ee6743e4d11fb9c6a1e6064b3693a1b600e7f5f5988047d98b3dc9fb90b"
|
||||
dependencies = [
|
||||
"libc",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "lazy_static"
|
||||
version = "1.4.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646"
|
||||
|
||||
[[package]]
|
||||
name = "libc"
|
||||
version = "0.2.139"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "201de327520df007757c1f0adce6e827fe8562fbc28bfd9c15571c66ca1f5f79"
|
||||
|
||||
[[package]]
|
||||
name = "libgit2-sys"
|
||||
version = "0.14.1+1.5.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "4a07fb2692bc3593bda59de45a502bb3071659f2c515e28c71e728306b038e17"
|
||||
dependencies = [
|
||||
"cc",
|
||||
"libc",
|
||||
"libssh2-sys",
|
||||
"libz-sys",
|
||||
"openssl-sys",
|
||||
"pkg-config",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "libssh2-sys"
|
||||
version = "0.2.23"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "b094a36eb4b8b8c8a7b4b8ae43b2944502be3e59cd87687595cf6b0a71b3f4ca"
|
||||
dependencies = [
|
||||
"cc",
|
||||
"libc",
|
||||
"libz-sys",
|
||||
"openssl-sys",
|
||||
"pkg-config",
|
||||
"vcpkg",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "libz-sys"
|
||||
version = "1.1.8"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "9702761c3935f8cc2f101793272e202c72b99da8f4224a19ddcf1279a6450bbf"
|
||||
dependencies = [
|
||||
"cc",
|
||||
"libc",
|
||||
"pkg-config",
|
||||
"vcpkg",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "log"
|
||||
version = "0.4.17"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "abb12e687cfb44aa40f41fc3978ef76448f9b6038cad6aef4259d3c095a2382e"
|
||||
dependencies = [
|
||||
"cfg-if",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "number_prefix"
|
||||
version = "0.4.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "830b246a0e5f20af87141b25c173cd1b609bd7779a4617d6ec582abaf90870f3"
|
||||
|
||||
[[package]]
|
||||
name = "openssl-probe"
|
||||
version = "0.1.5"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "ff011a302c396a5197692431fc1948019154afc178baf7d8e37367442a4601cf"
|
||||
|
||||
[[package]]
|
||||
name = "openssl-sys"
|
||||
version = "0.9.80"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "23bbbf7854cd45b83958ebe919f0e8e516793727652e27fda10a8384cfc790b7"
|
||||
dependencies = [
|
||||
"autocfg",
|
||||
"cc",
|
||||
"libc",
|
||||
"pkg-config",
|
||||
"vcpkg",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "percent-encoding"
|
||||
version = "2.2.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "478c572c3d73181ff3c2539045f6eb99e5491218eae919370993b890cdbdd98e"
|
||||
|
||||
[[package]]
|
||||
name = "pkg-config"
|
||||
version = "0.3.26"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "6ac9a59f73473f1b8d852421e59e64809f025994837ef743615c6d0c5b305160"
|
||||
|
||||
[[package]]
|
||||
name = "portable-atomic"
|
||||
version = "0.3.19"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "26f6a7b87c2e435a3241addceeeff740ff8b7e76b74c13bf9acb17fa454ea00b"
|
||||
|
||||
[[package]]
|
||||
name = "proc-macro2"
|
||||
version = "1.0.50"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "6ef7d57beacfaf2d8aee5937dab7b7f28de3cb8b1828479bb5de2a7106f2bae2"
|
||||
dependencies = [
|
||||
"unicode-ident",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "quote"
|
||||
version = "1.0.23"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "8856d8364d252a14d474036ea1358d63c9e6965c8e5c1885c18f73d70bff9c7b"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "redox_syscall"
|
||||
version = "0.2.16"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "fb5a58c1855b4b6819d59012155603f0b22ad30cad752600aadfcb695265519a"
|
||||
dependencies = [
|
||||
"bitflags",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "remove_dir_all"
|
||||
version = "0.5.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "3acd125665422973a33ac9d3dd2df85edad0f4ae9b00dafb1a05e43a9f5ef8e7"
|
||||
dependencies = [
|
||||
"winapi",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "rustversion"
|
||||
version = "1.0.11"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5583e89e108996506031660fe09baa5011b9dd0341b89029313006d1fb508d70"
|
||||
|
||||
[[package]]
|
||||
name = "serde"
|
||||
version = "1.0.152"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "bb7d1f0d3021d347a83e556fc4683dea2ea09d87bccdf88ff5c12545d89d5efb"
|
||||
dependencies = [
|
||||
"serde_derive",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "serde_derive"
|
||||
version = "1.0.152"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "af487d118eecd09402d70a5d72551860e788df87b464af30e5ea6a38c75c541e"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
"syn",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "shell-words"
|
||||
version = "1.1.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "24188a676b6ae68c3b2cb3a01be17fbf7240ce009799bb56d5b1409051e78fde"
|
||||
|
||||
[[package]]
|
||||
name = "strum"
|
||||
version = "0.24.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "063e6045c0e62079840579a7e47a355ae92f60eb74daaf156fb1e84ba164e63f"
|
||||
dependencies = [
|
||||
"strum_macros",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "strum_macros"
|
||||
version = "0.24.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "1e385be0d24f186b4ce2f9982191e7101bb737312ad61c1f2f984f34bcf85d59"
|
||||
dependencies = [
|
||||
"heck",
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
"rustversion",
|
||||
"syn",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "syn"
|
||||
version = "1.0.107"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "1f4064b5b16e03ae50984a5a8ed5d4f8803e6bc1fd170a3cda91a1be4b18e3f5"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
"unicode-ident",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "tempfile"
|
||||
version = "3.3.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5cdb1ef4eaeeaddc8fbd371e5017057064af0911902ef36b39801f67cc6d79e4"
|
||||
dependencies = [
|
||||
"cfg-if",
|
||||
"fastrand",
|
||||
"libc",
|
||||
"redox_syscall",
|
||||
"remove_dir_all",
|
||||
"winapi",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "tinyvec"
|
||||
version = "1.6.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "87cc5ceb3875bb20c2890005a4e226a4651264a5c75edb2421b52861a0a0cb50"
|
||||
dependencies = [
|
||||
"tinyvec_macros",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "tinyvec_macros"
|
||||
version = "0.1.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "cda74da7e1a664f795bb1f8a87ec406fb89a02522cf6e50620d016add6dbbf5c"
|
||||
|
||||
[[package]]
|
||||
name = "toml"
|
||||
version = "0.5.11"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "f4f7f0dd8d50a853a531c426359045b1998f04219d88799810762cd4ad314234"
|
||||
dependencies = [
|
||||
"serde",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "unicode-bidi"
|
||||
version = "0.3.9"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "0046be40136ef78dc325e0edefccf84ccddacd0afcc1ca54103fa3c61bbdab1d"
|
||||
|
||||
[[package]]
|
||||
name = "unicode-ident"
|
||||
version = "1.0.6"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "84a22b9f218b40614adcb3f4ff08b703773ad44fa9423e4e0d346d5db86e4ebc"
|
||||
|
||||
[[package]]
|
||||
name = "unicode-normalization"
|
||||
version = "0.1.22"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5c5713f0fc4b5db668a2ac63cdb7bb4469d8c9fed047b1d0292cc7b0ce2ba921"
|
||||
dependencies = [
|
||||
"tinyvec",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "unicode-width"
|
||||
version = "0.1.10"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "c0edd1e5b14653f783770bce4a4dabb4a5108a5370a5f5d8cfe8710c361f6c8b"
|
||||
|
||||
[[package]]
|
||||
name = "url"
|
||||
version = "2.3.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "0d68c799ae75762b8c3fe375feb6600ef5602c883c5d21eb51c09f22b83c4643"
|
||||
dependencies = [
|
||||
"form_urlencoded",
|
||||
"idna",
|
||||
"percent-encoding",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "vcpkg"
|
||||
version = "0.2.15"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "accd4ea62f7bb7a82fe23066fb0957d48ef677f6eeb8215f372f52e48bb32426"
|
||||
|
||||
[[package]]
|
||||
name = "winapi"
|
||||
version = "0.3.9"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419"
|
||||
dependencies = [
|
||||
"winapi-i686-pc-windows-gnu",
|
||||
"winapi-x86_64-pc-windows-gnu",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "winapi-i686-pc-windows-gnu"
|
||||
version = "0.4.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6"
|
||||
|
||||
[[package]]
|
||||
name = "winapi-x86_64-pc-windows-gnu"
|
||||
version = "0.4.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f"
|
||||
|
||||
[[package]]
|
||||
name = "windows-sys"
|
||||
version = "0.42.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5a3e1820f08b8513f676f7ab6c1f99ff312fb97b553d30ff4dd86f9f15728aa7"
|
||||
dependencies = [
|
||||
"windows_aarch64_gnullvm",
|
||||
"windows_aarch64_msvc",
|
||||
"windows_i686_gnu",
|
||||
"windows_i686_msvc",
|
||||
"windows_x86_64_gnu",
|
||||
"windows_x86_64_gnullvm",
|
||||
"windows_x86_64_msvc",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "windows_aarch64_gnullvm"
|
||||
version = "0.42.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "8c9864e83243fdec7fc9c5444389dcbbfd258f745e7853198f365e3c4968a608"
|
||||
|
||||
[[package]]
|
||||
name = "windows_aarch64_msvc"
|
||||
version = "0.42.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "4c8b1b673ffc16c47a9ff48570a9d85e25d265735c503681332589af6253c6c7"
|
||||
|
||||
[[package]]
|
||||
name = "windows_i686_gnu"
|
||||
version = "0.42.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "de3887528ad530ba7bdbb1faa8275ec7a1155a45ffa57c37993960277145d640"
|
||||
|
||||
[[package]]
|
||||
name = "windows_i686_msvc"
|
||||
version = "0.42.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "bf4d1122317eddd6ff351aa852118a2418ad4214e6613a50e0191f7004372605"
|
||||
|
||||
[[package]]
|
||||
name = "windows_x86_64_gnu"
|
||||
version = "0.42.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "c1040f221285e17ebccbc2591ffdc2d44ee1f9186324dd3e84e99ac68d699c45"
|
||||
|
||||
[[package]]
|
||||
name = "windows_x86_64_gnullvm"
|
||||
version = "0.42.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "628bfdf232daa22b0d64fdb62b09fcc36bb01f05a3939e20ab73aaf9470d0463"
|
||||
|
||||
[[package]]
|
||||
name = "windows_x86_64_msvc"
|
||||
version = "0.42.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "447660ad36a13288b1db4d4248e857b510e8c3a225c822ba4fb748c0aafecffd"
|
||||
|
||||
[[package]]
|
||||
name = "zeroize"
|
||||
version = "1.5.7"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "c394b5bd0c6f669e7275d9c20aa90ae064cb22e75a1cad54e1b34088034b149f"
|
15
Cargo.toml
Normal file
15
Cargo.toml
Normal file
@ -0,0 +1,15 @@
|
||||
[package]
|
||||
name = "dotfiles_installer"
|
||||
version = "0.1.0"
|
||||
edition = "2021"
|
||||
|
||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||
|
||||
[dependencies]
|
||||
dialoguer = "0.10.3"
|
||||
indicatif = "0.17.3"
|
||||
git2 = "0.16.0"
|
||||
strum = { version = "0.24.1", features = ["derive"] }
|
||||
serde = { version = "1.0.152", features = ["derive"] }
|
||||
toml = "0.5.10"
|
||||
fs_extra = "1.2.0"
|
35
src/config.rs
Normal file
35
src/config.rs
Normal file
@ -0,0 +1,35 @@
|
||||
use std::collections::HashMap;
|
||||
use serde::Deserialize;
|
||||
|
||||
#[derive(Clone, Debug, Deserialize)]
|
||||
pub struct Module {
|
||||
#[serde(default)]
|
||||
pub ignore: Vec<String>,
|
||||
pub content: HashMap<String, String>
|
||||
}
|
||||
|
||||
#[derive(Debug, Deserialize)]
|
||||
pub struct ModToml {
|
||||
pub modules: HashMap<String, Module>
|
||||
}
|
||||
|
||||
pub fn parse() -> Result<ModToml, String> {
|
||||
let mut files: ModToml = toml::from_str(std::fs::read_to_string("mod.toml").unwrap().as_str())
|
||||
.map_err(|e| format!("Error while parsing files.toml:\n{}", e.to_string()))?;
|
||||
|
||||
let home_path = std::env::var("HOME").unwrap();
|
||||
for (_, m) in files.modules.iter_mut() {
|
||||
m.content = m.content.iter().map(|(n, p)| (
|
||||
n.clone(),
|
||||
p.replace("~", home_path.as_str())
|
||||
)).collect()
|
||||
}
|
||||
|
||||
for (name, _path) in files.modules.iter().flat_map(|(_, m)| m.content.iter()) {
|
||||
let p = std::path::Path::new(name);
|
||||
if !p.exists() {
|
||||
return Err(format!("Missing module entry {}", name));
|
||||
}
|
||||
}
|
||||
Ok(files)
|
||||
}
|
71
src/main.rs
Normal file
71
src/main.rs
Normal file
@ -0,0 +1,71 @@
|
||||
mod prompt;
|
||||
mod repository;
|
||||
mod config;
|
||||
mod operations;
|
||||
mod utils;
|
||||
|
||||
use strum::{EnumIter, IntoEnumIterator};
|
||||
|
||||
#[derive(Debug, Clone, EnumIter)]
|
||||
enum MainMenu {
|
||||
INSTALL,
|
||||
COLLECT,
|
||||
DIFF,
|
||||
UPLOAD,
|
||||
QUIT
|
||||
}
|
||||
|
||||
impl ToString for MainMenu {
|
||||
fn to_string(&self) -> String {
|
||||
match self {
|
||||
MainMenu::INSTALL => "Pull and install files",
|
||||
MainMenu::COLLECT => "Collect files",
|
||||
MainMenu::DIFF => "View git diff",
|
||||
MainMenu::UPLOAD => "(Commit) and push to git",
|
||||
MainMenu::QUIT => "Exit"
|
||||
}.to_string()
|
||||
}
|
||||
}
|
||||
|
||||
fn main() {
|
||||
let repo = match repository::open_repo() {
|
||||
Ok(v) => v,
|
||||
Err(err) => {
|
||||
println!("Failed to open/clone repo!");
|
||||
println!("{}", err);
|
||||
return;
|
||||
}
|
||||
};
|
||||
|
||||
if !std::path::Path::new("mod.toml").exists() {
|
||||
println!("'mod.toml' doesn't exist!");
|
||||
return;
|
||||
}
|
||||
|
||||
let mods = match config::parse() {
|
||||
Ok(v) => v,
|
||||
Err(err) => {
|
||||
println!("Failed to process mod.toml!");
|
||||
println!("{}", err);
|
||||
return;
|
||||
}
|
||||
};
|
||||
|
||||
'main_loop: loop {
|
||||
let res = prompt::select(Some("What do you want to do?"), MainMenu::iter().collect());
|
||||
match res {
|
||||
MainMenu::INSTALL => {
|
||||
if let Err(err) = repository::pull_repo(&repo) {
|
||||
println!("Failed to pull repo!");
|
||||
println!("{}", err);
|
||||
return;
|
||||
}
|
||||
operations::install(&mods);
|
||||
},
|
||||
MainMenu::COLLECT => operations::collect(&mods),
|
||||
MainMenu::DIFF => { std::process::Command::new("git").arg("diff").spawn().unwrap().wait().unwrap(); },
|
||||
MainMenu::UPLOAD => operations::upload(&repo),
|
||||
MainMenu::QUIT => break 'main_loop
|
||||
}
|
||||
}
|
||||
}
|
113
src/operations.rs
Normal file
113
src/operations.rs
Normal file
@ -0,0 +1,113 @@
|
||||
use crate::config::{ModToml, Module};
|
||||
use crate::prompt::multi_select;
|
||||
|
||||
#[derive(Debug)]
|
||||
struct NamedModule(String, Module);
|
||||
impl ToString for NamedModule {
|
||||
fn to_string(&self) -> String {
|
||||
self.0.clone()
|
||||
}
|
||||
}
|
||||
|
||||
fn install_mod(m: NamedModule) {
|
||||
let spinner = indicatif::ProgressBar::new_spinner().with_message(format!("Installing module {}...", m.0));
|
||||
'content_iter: for entry in m.1.content {
|
||||
let src = std::path::Path::new(entry.0.as_str());
|
||||
if src.is_file() {
|
||||
if let Err(err) = std::fs::copy(src, entry.1) {
|
||||
println!("Failed to copy {}:\n{}", entry.0, err);
|
||||
}
|
||||
spinner.tick();
|
||||
} else {
|
||||
let dst = std::path::Path::new(entry.1.as_str());
|
||||
|
||||
if let Err(err) = std::fs::create_dir_all(dst) {
|
||||
println!("Failed to create directory {}:\n{}", entry.1, err);
|
||||
continue 'content_iter;
|
||||
}
|
||||
|
||||
let to_remove = match crate::utils::get_dir_content_filtered(dst, &m.1.ignore) {
|
||||
Ok(v) => v,
|
||||
Err(err) => { println!("Failed to get destination content:\n{}", err); continue 'content_iter; }
|
||||
};
|
||||
|
||||
for file in to_remove.files {
|
||||
let _ = std::fs::remove_file(file);
|
||||
spinner.tick();
|
||||
}
|
||||
|
||||
for dir in to_remove.directories.into_iter().rev() {
|
||||
let _ = std::fs::remove_dir(dir);
|
||||
spinner.tick();
|
||||
}
|
||||
if let Err(err) = crate::utils::copy_dir(src, dst, &spinner) {
|
||||
println!("Failed to copy directory:\n{}", err);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn collect_mod(m: NamedModule) {
|
||||
let spinner = indicatif::ProgressBar::new_spinner().with_message(format!("Collecting module {}...", m.0));
|
||||
'content_iter: for entry in m.1.content {
|
||||
let dst = std::path::Path::new(entry.0.as_str());
|
||||
let src = std::path::Path::new(entry.1.as_str());
|
||||
if src.is_file() {
|
||||
if let Err(err) = std::fs::copy(src, dst) {
|
||||
println!("Failed to copy {}:\n{}", entry.1, err);
|
||||
}
|
||||
spinner.tick();
|
||||
} else {
|
||||
let _ = std::fs::remove_dir_all(dst);
|
||||
if let Err(err) = std::fs::create_dir_all(dst) {
|
||||
println!("Failed to create directory {}:\n{}", entry.0, err);
|
||||
continue 'content_iter;
|
||||
}
|
||||
|
||||
let to_copy = match crate::utils::get_dir_content_filtered(src, &m.1.ignore) {
|
||||
Ok(v) => crate::utils::trim_dir_content(v, src),
|
||||
Err(err) => { println!("Failed to get source content:\n{}", err); continue 'content_iter; }
|
||||
};
|
||||
|
||||
for dir in to_copy.directories {
|
||||
std::fs::create_dir_all(dst.join(&dir)).unwrap();
|
||||
spinner.tick();
|
||||
}
|
||||
|
||||
for file in to_copy.files {
|
||||
std::fs::copy(src.join(&file), dst.join(&file)).unwrap();
|
||||
spinner.tick();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub fn install(config: &ModToml) {
|
||||
let to_install = multi_select(
|
||||
Some("Which modules do you want to install?"),
|
||||
config.modules.iter().map(|v| NamedModule(v.0.clone(), v.1.clone())).collect()
|
||||
);
|
||||
|
||||
for m in to_install {
|
||||
install_mod(m);
|
||||
}
|
||||
}
|
||||
|
||||
pub fn collect(config: &ModToml) {
|
||||
let to_collect = multi_select(
|
||||
Some("Which modules do you want to collect?"),
|
||||
config.modules.iter().map(|v| NamedModule(v.0.clone(), v.1.clone())).collect()
|
||||
);
|
||||
|
||||
for m in to_collect {
|
||||
collect_mod(m);
|
||||
}
|
||||
}
|
||||
|
||||
pub fn upload(repo: &git2::Repository) {
|
||||
if !crate::repository::is_clean(repo) {
|
||||
let msg = crate::prompt::input("Commit message");
|
||||
crate::repository::commit_changes(repo, msg);
|
||||
}
|
||||
crate::repository::push_repo(repo).expect("Failed to push to origin");
|
||||
}
|
38
src/prompt.rs
Normal file
38
src/prompt.rs
Normal file
@ -0,0 +1,38 @@
|
||||
use dialoguer::{Input, MultiSelect, Select};
|
||||
|
||||
pub fn select<F>(title: Option<&str>, mut options: Vec<F>) -> F
|
||||
where F: ToString
|
||||
{
|
||||
let mut sel = Select::new();
|
||||
sel.items(&options);
|
||||
sel.default(0);
|
||||
if let Some(title) = title {
|
||||
sel.with_prompt(title);
|
||||
}
|
||||
let sel = sel.interact().expect("User didn't enter anything");
|
||||
|
||||
options.remove(sel)
|
||||
}
|
||||
|
||||
pub fn multi_select<F>(title: Option<&str>, options: Vec<F>) -> Vec<F>
|
||||
where F: ToString
|
||||
{
|
||||
let mut sel = MultiSelect::new();
|
||||
sel.items(&options);
|
||||
if let Some(title) = title {
|
||||
sel.with_prompt(title);
|
||||
}
|
||||
let sel = sel.interact().expect("User didn't enter anything");
|
||||
|
||||
options.into_iter()
|
||||
.enumerate()
|
||||
.filter_map(|(i, v)| sel.contains(&i).then_some(v))
|
||||
.collect()
|
||||
}
|
||||
|
||||
pub fn input(title: &str) -> String {
|
||||
Input::new()
|
||||
.with_prompt(title)
|
||||
.interact_text()
|
||||
.expect("User didn't enter anything")
|
||||
}
|
131
src/repository.rs
Normal file
131
src/repository.rs
Normal file
@ -0,0 +1,131 @@
|
||||
use std::path::Path;
|
||||
use git2::Repository;
|
||||
|
||||
fn generate_callbacks(use_ssh_agent: bool, pb: &mut indicatif::ProgressBar) -> git2::RemoteCallbacks {
|
||||
let mut callbacks = git2::RemoteCallbacks::new();
|
||||
|
||||
if use_ssh_agent {
|
||||
callbacks.credentials(|_, username, _| git2::Cred::ssh_key_from_agent(username.unwrap()));
|
||||
} else {
|
||||
callbacks.credentials(|_, username, _| git2::Cred::ssh_key(
|
||||
username.unwrap(),
|
||||
None,
|
||||
Path::new(&format!("{}/.ssh/id_rsa", std::env::var("HOME").unwrap_or(String::new()))),
|
||||
None
|
||||
));
|
||||
}
|
||||
|
||||
callbacks.transfer_progress(|progress| {
|
||||
if progress.total_deltas() == 0 {
|
||||
pb.set_length(progress.total_objects().max(1) as u64);
|
||||
pb.set_position(progress.received_objects() as u64);
|
||||
pb.set_message("Receiving objects");
|
||||
} else {
|
||||
pb.set_length(progress.total_deltas().max(1) as u64);
|
||||
pb.set_position(progress.total_objects() as u64);
|
||||
pb.set_message("Resolving deltas");
|
||||
}
|
||||
true
|
||||
});
|
||||
callbacks.push_transfer_progress(|cur, tot, _| {
|
||||
pb.set_length(tot.max(1) as u64);
|
||||
pb.set_position(cur as u64);
|
||||
});
|
||||
|
||||
callbacks
|
||||
}
|
||||
|
||||
fn _clone_repo(callbacks: git2::RemoteCallbacks) -> Result<Repository, git2::Error> {
|
||||
let mut fo = git2::FetchOptions::new();
|
||||
fo.remote_callbacks(callbacks);
|
||||
|
||||
let mut builder = git2::build::RepoBuilder::new();
|
||||
builder.fetch_options(fo);
|
||||
|
||||
builder.clone(
|
||||
"git@ssh.gitlab.mattv.de:root/dotfiles.git",
|
||||
Path::new(".")
|
||||
)
|
||||
}
|
||||
|
||||
fn clone_repo() -> Result<Repository, git2::Error> {
|
||||
println!("Cloning repo...");
|
||||
let mut pb = indicatif::ProgressBar::new(1);
|
||||
pb.set_style(indicatif::ProgressStyle::with_template("{spinner} [{wide_bar}] {pos:>7}/{len:7} {msg}").unwrap().progress_chars("#>-"));
|
||||
_clone_repo(generate_callbacks(true, &mut pb))
|
||||
.or_else(|_| _clone_repo(generate_callbacks(false, &mut pb)))
|
||||
}
|
||||
|
||||
fn _pull_repo(repo: &Repository, callbacks: git2::RemoteCallbacks) -> Result<(), git2::Error> {
|
||||
let mut fo = git2::FetchOptions::new();
|
||||
fo.remote_callbacks(callbacks);
|
||||
|
||||
let mut co = git2::build::CheckoutBuilder::new();
|
||||
co.force();
|
||||
|
||||
repo.find_remote("origin").unwrap().fetch(&["main"], Some(&mut fo), None)?;
|
||||
|
||||
let mut commits = vec![];
|
||||
repo.fetchhead_foreach(|name, url, oid, was_merge| if was_merge { commits.push(repo.annotated_commit_from_fetchhead(name, &String::from_utf8_lossy(url), oid)); true} else { true })?;
|
||||
let commits = commits.into_iter().collect::<Result<Vec<_>, git2::Error>>()?;
|
||||
if commits.is_empty() {
|
||||
panic!("Failed to get heads from origin!");
|
||||
}
|
||||
|
||||
repo.merge(commits.iter().collect::<Vec<_>>().as_slice(), None, Some(&mut co))
|
||||
}
|
||||
|
||||
pub fn pull_repo(repo: &Repository) -> Result<(), git2::Error> {
|
||||
println!("Pulling repo...");
|
||||
let mut pb = indicatif::ProgressBar::new(1);
|
||||
pb.set_style(indicatif::ProgressStyle::with_template("{spinner} [{wide_bar}] {pos:>7}/{len:7} {msg}").unwrap().progress_chars("#>-"));
|
||||
_pull_repo(repo, generate_callbacks(true, &mut pb))
|
||||
.or_else(|_| _pull_repo(repo, generate_callbacks(false, &mut pb)))
|
||||
}
|
||||
|
||||
fn _push_repo(repo: &Repository, callbacks: git2::RemoteCallbacks) -> Result<(), git2::Error> {
|
||||
let mut pu = git2::PushOptions::new();
|
||||
pu.remote_callbacks(callbacks);
|
||||
|
||||
let mut remote = repo.find_remote("origin").unwrap();
|
||||
|
||||
remote.push::<&str>(&["refs/heads/main:refs/heads/main"], Some(&mut pu))
|
||||
}
|
||||
|
||||
pub fn push_repo(repo: &Repository) -> Result<(), git2::Error> {
|
||||
println!("Pushing repo...");
|
||||
let mut pb = indicatif::ProgressBar::new(1);
|
||||
pb.set_style(indicatif::ProgressStyle::with_template("{spinner} [{wide_bar}] {bytes}/{total_bytes} {msg}").unwrap().progress_chars("#>-"));
|
||||
_push_repo(repo, generate_callbacks(true, &mut pb))
|
||||
.or_else(|_| _push_repo(repo, generate_callbacks(false, &mut pb)))
|
||||
}
|
||||
|
||||
pub fn open_repo() -> Result<Repository, String> {
|
||||
Repository::open(".")
|
||||
.or_else(|_| clone_repo())
|
||||
.map_err(|e| e.message().to_string())
|
||||
}
|
||||
|
||||
pub fn is_clean(repo: &Repository) -> bool {
|
||||
let mut so = git2::StatusOptions::new();
|
||||
so.show(git2::StatusShow::IndexAndWorkdir);
|
||||
so.include_untracked(true);
|
||||
so.include_ignored(false);
|
||||
so.include_unmodified(false);
|
||||
|
||||
repo.statuses(Some(&mut so)).map_or(false, |s| s.is_empty())
|
||||
}
|
||||
|
||||
pub fn commit_changes(repo: &Repository, msg: String) {
|
||||
let sig = git2::Signature::now("Mutzi", "root@mattv.de").unwrap();
|
||||
|
||||
let mut index = repo.index().unwrap();
|
||||
index.add_all(["*"], git2::IndexAddOption::DEFAULT, None).unwrap();
|
||||
|
||||
let tree_id = index.write_tree().unwrap();
|
||||
let tree = repo.find_tree(tree_id).unwrap();
|
||||
let parent_id = repo.refname_to_id("HEAD").unwrap();
|
||||
let parent = repo.find_commit(parent_id).unwrap();
|
||||
|
||||
repo.commit(Some("HEAD"), &sig, &sig, msg.as_str(), &tree, &[&parent]).unwrap();
|
||||
}
|
47
src/utils.rs
Normal file
47
src/utils.rs
Normal file
@ -0,0 +1,47 @@
|
||||
use fs_extra::dir::DirContent;
|
||||
|
||||
pub fn get_dir_content_filtered(p: &std::path::Path, ignored: &Vec<String>) -> Result<DirContent, String> {
|
||||
let p = p.canonicalize().map_err(|e| e.to_string())?;
|
||||
let base_path_len = p.to_str().unwrap().len() + 1;
|
||||
|
||||
let retain_fn = |p: &String| {
|
||||
if p.len() < base_path_len { return false; }
|
||||
let rel_path = p.split_at(base_path_len).1;
|
||||
!ignored.iter().any(|i| rel_path.starts_with(i))
|
||||
};
|
||||
|
||||
let mut content = fs_extra::dir::get_dir_content(p).map_err(|e| e.to_string())?;
|
||||
content.files.retain(retain_fn);
|
||||
content.directories.retain(retain_fn);
|
||||
Ok(content)
|
||||
}
|
||||
|
||||
pub fn trim_dir_content(mut content: DirContent, base: &std::path::Path) -> DirContent {
|
||||
let base_path_len = base.canonicalize().unwrap().to_str().unwrap().len() + 1;
|
||||
|
||||
let iter_fn = |p: &mut String| {
|
||||
if p.len() < base_path_len { return; }
|
||||
let _ = p.drain(..base_path_len).collect::<Vec<_>>();
|
||||
};
|
||||
|
||||
content.files.iter_mut().for_each(iter_fn);
|
||||
content.directories.iter_mut().for_each(iter_fn);
|
||||
content
|
||||
}
|
||||
|
||||
pub fn copy_dir(src_dir: &std::path::Path, dst_dir: &std::path::Path, spinner: &indicatif::ProgressBar) -> Result<(), String> {
|
||||
let content = src_dir.read_dir().map_err(|e| e.to_string())?;
|
||||
for entry in content {
|
||||
let entry = entry.map_err(|e| e.to_string())?;
|
||||
let new_dst = dst_dir.join(entry.file_name());
|
||||
if entry.path().is_file() {
|
||||
std::fs::copy(entry.path(), new_dst).map_err(|e| e.to_string())?;
|
||||
spinner.tick();
|
||||
} else {
|
||||
std::fs::create_dir_all(&new_dst).map_err(|e| e.to_string())?;
|
||||
spinner.tick();
|
||||
copy_dir(&entry.path(), &new_dst, spinner)?;
|
||||
}
|
||||
}
|
||||
Ok(())
|
||||
}
|
Loading…
Reference in New Issue
Block a user