feat: managed zsh config + optional installers + version bump action

This commit is contained in:
krol
2026-02-06 21:17:58 +03:00
parent 0ef50a858f
commit 569f3d6538
7 changed files with 462 additions and 151 deletions

67
.github/workflows/bump-version.yml vendored Normal file
View File

@@ -0,0 +1,67 @@
name: Bump version.txt
on:
push:
branches: [ "main" ]
# Avoid infinite loop when this workflow commits version.txt.
paths-ignore:
- "version.txt"
concurrency:
group: bump-version-${{ github.ref }}
cancel-in-progress: true
permissions:
contents: write
jobs:
bump:
if: github.actor != 'github-actions[bot]'
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v4
with:
fetch-depth: 0
- name: Bump patch version in version.txt
shell: bash
run: |
set -euo pipefail
file="version.txt"
if [[ ! -f "$file" ]]; then
echo "0.0.1" > "$file"
else
v="$(tr -d '[:space:]' < "$file")"
if [[ "$v" =~ ^([0-9]+)\.([0-9]+)\.([0-9]+)$ ]]; then
major="${BASH_REMATCH[1]}"
minor="${BASH_REMATCH[2]}"
patch="${BASH_REMATCH[3]}"
patch="$((patch + 1))"
echo "${major}.${minor}.${patch}" > "$file"
else
echo "version.txt has invalid format: '$v' (expected x.y.z)" >&2
exit 1
fi
fi
echo "New version: $(cat "$file")"
- name: Commit and push
shell: bash
run: |
set -euo pipefail
git config user.name "github-actions[bot]"
git config user.email "github-actions[bot]@users.noreply.github.com"
# If version didn't change for any reason, do nothing.
git add version.txt
if git diff --cached --quiet; then
echo "No changes to commit."
exit 0
fi
git commit -m "chore: bump version [skip ci]"
git push

275
.zshrc
View File

@@ -1,50 +1,75 @@
# --- zshrc auto-updater (github.com/ushst/my-zsh) --- # --- my-zsh loader + auto-updater (github.com/ushst/my-zsh) ---
typeset -gA UpdaterCfg # This file is intended to stay small and stable.
UpdaterCfg[repoRawBase]="https://raw.githubusercontent.com/ushst/my-zsh/main" # Your personal config should go to ~/.zshrc.local or ~/.zshrc.d/*.zsh
UpdaterCfg[remoteZshrcPath]=".zshrc"
UpdaterCfg[remoteVersionPath]="version.txt"
UpdaterCfg[checkIntervalSeconds]=21600 # 6 часов
UpdaterCfg[offlineRetrySeconds]=300 # повторная попытка через 5 минут, если сети нет
UpdaterCfg[networkCheckRetries]=3
UpdaterCfg[networkRetryDelaySeconds]=1
UpdaterCfg[networkCheckTimeoutSeconds]=2
localZshrc="${HOME}/.zshrc" typeset -gA MyZshUpdaterCfg
localStateDir="${XDG_STATE_HOME:-${HOME}/.local/state}/zsh-updater" MyZshUpdaterCfg[repoRawBase]="https://raw.githubusercontent.com/ushst/my-zsh/main"
mkdir -p "${localStateDir}" MyZshUpdaterCfg[remoteVersionPath]="version.txt"
MyZshUpdaterCfg[remoteManagedPath]="zshrc.managed"
MyZshUpdaterCfg[checkIntervalSeconds]=21600 # 6 hours
MyZshUpdaterCfg[offlineRetrySeconds]=300 # retry in 5 minutes if offline
MyZshUpdaterCfg[networkCheckRetries]=3
MyZshUpdaterCfg[networkRetryDelaySeconds]=1
MyZshUpdaterCfg[networkCheckTimeoutSeconds]=2
lastCheckFile="${localStateDir}/last_check" my_zsh_state_dir="${XDG_STATE_HOME:-${HOME}/.local/state}/my-zsh"
localVersionFile="${localStateDir}/local_version" my_zsh_config_dir="${XDG_CONFIG_HOME:-${HOME}/.config}/my-zsh"
lockFile="${localStateDir}/lock" mkdir -p "${my_zsh_state_dir}" "${my_zsh_config_dir}" 2>/dev/null || true
deferredFlag="${localStateDir}/deferred_check_scheduled"
get_local_version() { my_zsh_last_check_file="${my_zsh_state_dir}/last_check"
[[ -f "${localVersionFile}" ]] && cat "${localVersionFile}" && return my_zsh_local_version_file="${my_zsh_state_dir}/local_version"
my_zsh_lock_dir="${my_zsh_state_dir}/lockdir"
my_zsh_deferred_flag="${my_zsh_state_dir}/deferred_check_scheduled"
my_zsh_managed_file="${my_zsh_config_dir}/zshrc.managed"
my_zsh_local_file="${HOME}/.zshrc.local"
my_zsh_local_dir="${HOME}/.zshrc.d"
my_zsh_get_local_version() {
[[ -f "${my_zsh_local_version_file}" ]] && cat "${my_zsh_local_version_file}" && return
echo "0.0.0" echo "0.0.0"
} }
version_lt() { [[ "$(printf '%s\n%s\n' "$1" "$2" | sort -V | head -n1)" == "$1" && "$1" != "$2" ]]; } my_zsh_version_lt() {
[[ "$(printf '%s\n%s\n' "$1" "$2" | sort -V | head -n1)" == "$1" && "$1" != "$2" ]]
should_check_now() {
now=$(date +%s)
[[ ! -f "${lastCheckFile}" ]] && return 0
last=$(<"${lastCheckFile}")
(( now - last >= UpdaterCfg[checkIntervalSeconds] ))
} }
with_lock() { exec 9>"${lockFile}"; flock -n 9 || return 1; "$@"; } my_zsh_should_check_now() {
local now last
now=$(date +%s)
[[ ! -f "${my_zsh_last_check_file}" ]] && return 0
last=$(<"${my_zsh_last_check_file}")
(( now - last >= MyZshUpdaterCfg[checkIntervalSeconds] ))
}
update_local_version_cache() { echo "$1" > "${localVersionFile}"; } my_zsh_with_lock() {
# Portable lock (no flock dependency): mkdir is atomic on POSIX filesystems.
if mkdir "${my_zsh_lock_dir}" 2>/dev/null; then
"$@"
rmdir "${my_zsh_lock_dir}" 2>/dev/null || true
return 0
fi
return 1
}
network_available() { my_zsh_network_available() {
local testUrl="${UpdaterCfg[repoRawBase]}/${UpdaterCfg[remoteVersionPath]}" local testUrl retries delay timeout attempt
local retries=${UpdaterCfg[networkCheckRetries]} testUrl="${MyZshUpdaterCfg[repoRawBase]}/${MyZshUpdaterCfg[remoteVersionPath]}"
local delay=${UpdaterCfg[networkRetryDelaySeconds]} retries=${MyZshUpdaterCfg[networkCheckRetries]}
local timeout=${UpdaterCfg[networkCheckTimeoutSeconds]} delay=${MyZshUpdaterCfg[networkRetryDelaySeconds]}
timeout=${MyZshUpdaterCfg[networkCheckTimeoutSeconds]}
for (( attempt = 1; attempt <= retries; attempt++ )); do for (( attempt = 1; attempt <= retries; attempt++ )); do
if curl -fsI --max-time "${timeout}" "${testUrl}" >/dev/null 2>&1; then if command -v curl >/dev/null 2>&1; then
return 0 if curl -fsI --max-time "${timeout}" "${testUrl}" >/dev/null 2>&1; then
return 0
fi
elif command -v wget >/dev/null 2>&1; then
if wget -q --spider --timeout="${timeout}" "${testUrl}" >/dev/null 2>&1; then
return 0
fi
else
return 1
fi fi
sleep "${delay}" sleep "${delay}"
done done
@@ -52,139 +77,109 @@ network_available() {
return 1 return 1
} }
schedule_offline_retry() { my_zsh_update_local_version_cache() { echo "$1" > "${my_zsh_local_version_file}"; }
local delay="${UpdaterCfg[offlineRetrySeconds]}"
[[ -f "${deferredFlag}" ]] && return 0
echo "${delay}" > "${deferredFlag}" my_zsh_schedule_offline_retry() {
local delay="${MyZshUpdaterCfg[offlineRetrySeconds]}"
[[ -f "${my_zsh_deferred_flag}" ]] && return 0
echo "${delay}" > "${my_zsh_deferred_flag}"
( (
sleep "${delay}" sleep "${delay}"
rm -f "${deferredFlag}" rm -f "${my_zsh_deferred_flag}"
zshrc_update_check 1 my_zsh_update_check 1
) >/dev/null 2>&1 & ) >/dev/null 2>&1 &
} }
zshrc_update_check() { my_zsh_fetch_to() {
local force="${1:-0}" local url out
url="$1"
(( force )) || should_check_now || return 0 out="$2"
if command -v curl >/dev/null 2>&1; then
if ! network_available; then curl -fsSL "${url}" -o "${out}"
echo "$(date +%s)" > "${lastCheckFile}" return $?
[[ -t 1 ]] && echo "[zsh-updater] Сети нет. Повторим проверку позже."
schedule_offline_retry
return 0
fi fi
if command -v wget >/dev/null 2>&1; then
with_lock _do_update_check wget -q -O "${out}" "${url}"
echo "$(date +%s)" > "${lastCheckFile}" return $?
fi
return 1
} }
_do_update_check() { my_zsh_do_update_check() {
local repo="${UpdaterCfg[repoRawBase]}" local repo remoteVersion localVersion tmpNew backup
local rverPath="${UpdaterCfg[remoteVersionPath]}"
local rcfgPath="${UpdaterCfg[remoteZshrcPath]}"
remoteVersion="$(curl -fsSL "${repo}/${rverPath}" 2>/dev/null || true)" repo="${MyZshUpdaterCfg[repoRawBase]}"
remoteVersion="$( (command -v curl >/dev/null 2>&1 && curl -fsSL "${repo}/${MyZshUpdaterCfg[remoteVersionPath]}" 2>/dev/null) || (command -v wget >/dev/null 2>&1 && wget -q -O - "${repo}/${MyZshUpdaterCfg[remoteVersionPath]}" 2>/dev/null) || true )"
[[ -n "${remoteVersion}" ]] || return 0 [[ -n "${remoteVersion}" ]] || return 0
localVersion="$(get_local_version)" localVersion="$(my_zsh_get_local_version)"
if version_lt "${localVersion}" "${remoteVersion}"; then # If managed file doesn't exist yet, treat it like version 0.0.0.
tmpNew="${localStateDir}/.zshrc.new" if [[ ! -f "${my_zsh_managed_file}" ]]; then
backup="${localStateDir}/zshrc.backup.$(date +%Y%m%d-%H%M%S)" localVersion="0.0.0"
fi
if my_zsh_version_lt "${localVersion}" "${remoteVersion}"; then
tmpNew="${my_zsh_state_dir}/zshrc.managed.new"
backup="${my_zsh_state_dir}/managed.backup.$(date +%Y%m%d-%H%M%S)"
# простая «полоса проверки»
if [[ -t 1 ]]; then if [[ -t 1 ]]; then
echo -n "[zsh-updater] Проверка" echo -n "[my-zsh] Checking updates"
for i in 1 2 3; do sleep 0.2; echo -n "."; done for i in 1 2 3; do sleep 0.2; echo -n "."; done
echo echo
fi fi
if curl -fsSL "${repo}/${rcfgPath}" -o "${tmpNew}"; then if my_zsh_fetch_to "${repo}/${MyZshUpdaterCfg[remoteManagedPath]}" "${tmpNew}"; then
cp -f "${localZshrc}" "${backup}" 2>/dev/null || true cp -f "${my_zsh_managed_file}" "${backup}" 2>/dev/null || true
mv -f "${tmpNew}" "${localZshrc}" mv -f "${tmpNew}" "${my_zsh_managed_file}"
update_local_version_cache "${remoteVersion}" my_zsh_update_local_version_cache "${remoteVersion}"
[[ -t 1 ]] && echo "[zsh-updater] Обновлено до ${remoteVersion}. Бэкап: ${backup}. Перезапусти shell." [[ -t 1 ]] && echo "[my-zsh] Updated managed config to ${remoteVersion}. Backup: ${backup}. Restart shell."
fi fi
fi fi
} }
trigger_zshrc_update_check() { my_zsh_update_check() {
# Проверяем в фоне, чтобы запуск shell не блокировался при отсутствии сети local force="${1:-0}"
( zshrc_update_check "$@" ) &!
}
trigger_zshrc_update_check # Allow disabling auto-update.
# --- end zshrc auto-updater --- [[ "${MY_ZSH_AUTOUPDATE:-1}" == "0" ]] && return 0
# If you come from bash you might have to change your $PATH. (( force )) || my_zsh_should_check_now || return 0
# export PATH=$HOME/bin:/usr/local/bin:$PATH
# Path to your oh-my-zsh installation. if ! my_zsh_network_available; then
export ZSH="$HOME/.oh-my-zsh" echo "$(date +%s)" > "${my_zsh_last_check_file}"
[[ -t 1 ]] && echo "[my-zsh] Offline. Will retry later."
# Theme my_zsh_schedule_offline_retry
ZSH_THEME="avit" return 0
# Plugins
plugins=(
git
zsh-autosuggestions
zsh-syntax-highlighting
)
source $ZSH/oh-my-zsh.sh
# User configuration
prompt_context() {
if [[ "$USER" != "$DEFAULT_USER" || -n "$SSH_CLIENT" ]]; then
prompt_segment black default "%(!.%{%F{yellow}%}.)"
fi fi
my_zsh_with_lock my_zsh_do_update_check || true
echo "$(date +%s)" > "${my_zsh_last_check_file}"
} }
export NVM_DIR="$HOME/.nvm" my_zsh_trigger_update_check() {
[ -s "$NVM_DIR/nvm.sh" ] && \. "$NVM_DIR/nvm.sh" # Run in background to avoid slowing shell startup.
[ -s "$NVM_DIR/bash_completion" ] && \. "$NVM_DIR/bash_completion" ( my_zsh_update_check "$@" ) &!
# Custom aliases and functions
alias untargz='tar -xvzf'
function a() {
case "$1" in
i)
shift
sudo apt install "$@"
;;
u)
sudo apt update
;;
up)
sudo apt upgrade -y
;;
all)
sudo apt update && sudo apt upgrade -y
;;
*)
echo "Неизвестная команда: a $1"
;;
esac
} }
extract() { # --- load managed config (auto-updated) ---
if [ -f "$1" ]; then if [[ -f "${my_zsh_managed_file}" ]]; then
case "$1" in source "${my_zsh_managed_file}"
*.tar.bz2) tar xvjf "$1" ;; fi
*.tar.gz) tar xvzf "$1" ;;
*.tar.xz) tar xvJf "$1" ;; # --- load user overrides ---
*.tar) tar xvf "$1" ;; if [[ -f "${my_zsh_local_file}" ]]; then
*.gz) gunzip "$1" ;; source "${my_zsh_local_file}"
*.zip) unzip "$1" ;; fi
*.rar) unrar x "$1" ;;
*.7z) 7z x "$1" ;; if [[ -d "${my_zsh_local_dir}" ]]; then
*) echo "Неизвестный формат: $1" ;; for f in "${my_zsh_local_dir}"/*.zsh(N); do
esac source "$f"
else done
echo "Файл не найден: $1" fi
fi
} # Update last: takes effect next shell.
alias x='extract' my_zsh_trigger_update_check
# --- end my-zsh loader + auto-updater ---

View File

@@ -3,9 +3,34 @@
`bash -c "$(curl -fsSL https://raw.githubusercontent.com/ushst/my-zsh/main/termux_zsh.sh)"` `bash -c "$(curl -fsSL https://raw.githubusercontent.com/ushst/my-zsh/main/termux_zsh.sh)"`
## Как теперь хранить свои изменения
- `~/.zshrc` это загрузчик + автообновление managed-конфига.
- Основная конфигурация обновляется в `~/.config/my-zsh/zshrc.managed`.
- Твои личные настройки добавляй в `~/.zshrc.local` или в файлы `~/.zshrc.d/*.zsh` (они не будут перезаписываться).
- Установщик ставит `~/.zshrc` один раз. Повторный запуск `install.sh`/`termux_zsh.sh` не перезапишет его, если там уже my-zsh loader.
## Отключить автообновление
Добавь в `~/.zshrc.local`:
`export MY_ZSH_AUTOUPDATE=0`
# MICRO
Конфиг micro лежит в репо: `.microrc` и `.config/micro/syntax/shell.yaml`.
Установить вручную:
`wget -O ~/.microrc https://raw.githubusercontent.com/ushst/my-zsh/main/.microrc`
`mkdir -p ~/.config/micro/syntax && wget -O ~/.config/micro/syntax/shell.yaml https://raw.githubusercontent.com/ushst/my-zsh/main/.config/micro/syntax/shell.yaml`
Или через установщик: он спросит, ставить ли micro-конфиг.
Можно без вопросов:
`MY_ZSH_INSTALL_MICRO=1 bash -c "$(curl -fsSL https://raw.githubusercontent.com/ushst/my-zsh/main/install.sh)"`
# NANO # NANO
`sudo wget -O ~/.nanorc https://raw.githubusercontent.com/ushst/my-zsh/main/.nanorc` `sudo wget -O ~/.nanorc https://raw.githubusercontent.com/ushst/my-zsh/main/.nanorc`
# MSFCONSOLE # MSFCONSOLE
`mkdir -p ~/.msf4 && wget -O ~/.msf4/msfconsole.rc https://raw.githubusercontent.com/ushst/my-zsh/main/msfconsole.rc` `mkdir -p ~/.msf4 && wget -O ~/.msf4/msfconsole.rc https://raw.githubusercontent.com/ushst/my-zsh/main/msfconsole.rc`
Или через установщик: он спросит, ставить ли msfconsole-конфиг.
Можно без вопросов:
`MY_ZSH_INSTALL_MSFCONSOLE=1 bash -c "$(curl -fsSL https://raw.githubusercontent.com/ushst/my-zsh/main/install.sh)"`

View File

@@ -5,6 +5,7 @@ set -e
ZSH_DIR="${HOME}/.oh-my-zsh" ZSH_DIR="${HOME}/.oh-my-zsh"
ZDOTDIR="${HOME}" ZDOTDIR="${HOME}"
ZSHRC_URL="https://raw.githubusercontent.com/ushst/my-zsh/main/.zshrc" ZSHRC_URL="https://raw.githubusercontent.com/ushst/my-zsh/main/.zshrc"
ZSHRC_MANAGED_URL="https://raw.githubusercontent.com/ushst/my-zsh/main/zshrc.managed"
PLUGINS_REPO1="https://github.com/zsh-users/zsh-autosuggestions" PLUGINS_REPO1="https://github.com/zsh-users/zsh-autosuggestions"
PLUGINS_REPO2="https://github.com/zsh-users/zsh-syntax-highlighting.git" PLUGINS_REPO2="https://github.com/zsh-users/zsh-syntax-highlighting.git"
# ============================= # =============================
@@ -23,8 +24,81 @@ echo "[*] Установка плагинов..."
git clone "$PLUGINS_REPO1" ${ZSH_CUSTOM:-~/.oh-my-zsh/custom}/plugins/zsh-autosuggestions || true git clone "$PLUGINS_REPO1" ${ZSH_CUSTOM:-~/.oh-my-zsh/custom}/plugins/zsh-autosuggestions || true
git clone "$PLUGINS_REPO2" ${ZSH_CUSTOM:-~/.oh-my-zsh/custom}/plugins/zsh-syntax-highlighting || true git clone "$PLUGINS_REPO2" ${ZSH_CUSTOM:-~/.oh-my-zsh/custom}/plugins/zsh-syntax-highlighting || true
echo "[*] Загрузка конфигурации .zshrc..." echo "[*] Установка загрузчика .zshrc (my-zsh)..."
wget -O "${ZDOTDIR}/.zshrc" "$ZSHRC_URL" # Ставим loader один раз: если он уже установлен, не перезаписываем ~/.zshrc.
if [ -f "${ZDOTDIR}/.zshrc" ] && grep -q "my-zsh loader + auto-updater" "${ZDOTDIR}/.zshrc" 2>/dev/null; then
echo "[*] ~/.zshrc уже установлен как my-zsh loader — пропускаю."
else
# Не теряем существующий ~/.zshrc: делаем бэкап, если он есть.
if [ -f "${ZDOTDIR}/.zshrc" ]; then
cp -f "${ZDOTDIR}/.zshrc" "${ZDOTDIR}/.zshrc.backup.$(date +%Y%m%d-%H%M%S)" || true
fi
wget -O "${ZDOTDIR}/.zshrc" "$ZSHRC_URL"
fi
echo "[*] Загрузка managed-конфига (auto-updated)..."
mkdir -p "${HOME}/.config/my-zsh" || true
wget -O "${HOME}/.config/my-zsh/zshrc.managed" "$ZSHRC_MANAGED_URL"
echo "[*] Создаю файл для пользовательских правок: ~/.zshrc.local (если его нет)..."
if [ ! -f "${HOME}/.zshrc.local" ]; then
cat > "${HOME}/.zshrc.local" <<'EOF'
# Your custom zsh config goes here.
# This file will NOT be overwritten by my-zsh updates.
EOF
fi
# ========= Optional configs =========
ask_yes_no() {
# Usage: ask_yes_no "Question" "N|Y"
local question default reply
question="$1"
default="${2:-N}"
# Non-interactive shell: honor default.
if [ ! -t 0 ]; then
[ "${default}" = "Y" ] && return 0 || return 1
fi
read -r -p "${question} [y/N]: " reply
reply="${reply:-$default}"
case "$reply" in
y|Y|yes|YES) return 0 ;;
*) return 1 ;;
esac
}
backup_if_exists() {
local p ts
p="$1"
ts="$(date +%Y%m%d-%H%M%S)"
if [ -f "$p" ]; then
cp -f "$p" "$p.backup.$ts" 2>/dev/null || true
fi
}
# Micro
MICRORC_URL="https://raw.githubusercontent.com/ushst/my-zsh/main/.microrc"
MICRO_SHELL_SYNTAX_URL="https://raw.githubusercontent.com/ushst/my-zsh/main/.config/micro/syntax/shell.yaml"
if [ "${MY_ZSH_INSTALL_MICRO:-}" = "1" ] || ( [ "${MY_ZSH_INSTALL_MICRO:-}" != "0" ] && ask_yes_no "[?] Установить конфиг для micro ( .microrc + syntax )?" "N" ); then
echo "[*] Установка конфига micro..."
backup_if_exists "${HOME}/.microrc"
wget -O "${HOME}/.microrc" "$MICRORC_URL"
mkdir -p "${HOME}/.config/micro/syntax" || true
backup_if_exists "${HOME}/.config/micro/syntax/shell.yaml"
wget -O "${HOME}/.config/micro/syntax/shell.yaml" "$MICRO_SHELL_SYNTAX_URL"
fi
# msfconsole
MSFCONSOLE_RC_URL="https://raw.githubusercontent.com/ushst/my-zsh/main/msfconsole.rc"
if [ "${MY_ZSH_INSTALL_MSFCONSOLE:-}" = "1" ] || ( [ "${MY_ZSH_INSTALL_MSFCONSOLE:-}" != "0" ] && ask_yes_no "[?] Установить конфиг для msfconsole ( alias s -> search )?" "N" ); then
echo "[*] Установка конфига msfconsole..."
mkdir -p "${HOME}/.msf4" || true
backup_if_exists "${HOME}/.msf4/msfconsole.rc"
wget -O "${HOME}/.msf4/msfconsole.rc" "$MSFCONSOLE_RC_URL"
fi
echo "[*] Смена стандартной оболочки на zsh..." echo "[*] Смена стандартной оболочки на zsh..."
chsh -s "$(which zsh)" "$USER" chsh -s "$(which zsh)" "$USER"

View File

@@ -5,6 +5,7 @@ set -e
ZSH_DIR="${HOME}/.oh-my-zsh" ZSH_DIR="${HOME}/.oh-my-zsh"
ZDOTDIR="${HOME}" ZDOTDIR="${HOME}"
ZSHRC_URL="https://raw.githubusercontent.com/ushst/my-zsh/main/.zshrc" ZSHRC_URL="https://raw.githubusercontent.com/ushst/my-zsh/main/.zshrc"
ZSHRC_MANAGED_URL="https://raw.githubusercontent.com/ushst/my-zsh/main/zshrc.managed"
PLUGINS_REPO1="https://github.com/zsh-users/zsh-autosuggestions" PLUGINS_REPO1="https://github.com/zsh-users/zsh-autosuggestions"
PLUGINS_REPO2="https://github.com/zsh-users/zsh-syntax-highlighting.git" PLUGINS_REPO2="https://github.com/zsh-users/zsh-syntax-highlighting.git"
# ============================= # =============================
@@ -23,22 +24,95 @@ echo "[*] Установка плагинов..."
git clone "$PLUGINS_REPO1" ${ZSH_CUSTOM:-~/.oh-my-zsh/custom}/plugins/zsh-autosuggestions || true git clone "$PLUGINS_REPO1" ${ZSH_CUSTOM:-~/.oh-my-zsh/custom}/plugins/zsh-autosuggestions || true
git clone "$PLUGINS_REPO2" ${ZSH_CUSTOM:-~/.oh-my-zsh/custom}/plugins/zsh-syntax-highlighting || true git clone "$PLUGINS_REPO2" ${ZSH_CUSTOM:-~/.oh-my-zsh/custom}/plugins/zsh-syntax-highlighting || true
echo "[*] Загрузка конфигурации .zshrc..." echo "[*] Установка загрузчика .zshrc (my-zsh)..."
wget -O "${ZDOTDIR}/.zshrc" "$ZSHRC_URL" # Ставим loader один раз: если он уже установлен, не перезаписываем ~/.zshrc.
if [ -f "${ZDOTDIR}/.zshrc" ] && grep -q "my-zsh loader + auto-updater" "${ZDOTDIR}/.zshrc" 2>/dev/null; then
echo "[*] ~/.zshrc уже установлен как my-zsh loader — пропускаю."
else
# Не теряем существующий ~/.zshrc: делаем бэкап, если он есть.
if [ -f "${ZDOTDIR}/.zshrc" ]; then
cp -f "${ZDOTDIR}/.zshrc" "${ZDOTDIR}/.zshrc.backup.$(date +%Y%m%d-%H%M%S)" || true
fi
wget -O "${ZDOTDIR}/.zshrc" "$ZSHRC_URL"
fi
# Если установка проходит в Termux — заменим "sudo apt install" на "apt install" в .zshrc echo "[*] Загрузка managed-конфига (auto-updated)..."
mkdir -p "${HOME}/.config/my-zsh" || true
wget -O "${HOME}/.config/my-zsh/zshrc.managed" "$ZSHRC_MANAGED_URL"
# Если установка проходит в Termux — заменим "sudo apt install" на "apt install" в managed-конфиге
if [ -n "$TERMUX_VERSION" ] || ( [ -n "$PREFIX" ] && [[ "$PREFIX" == *"com.termux"* ]] ); then if [ -n "$TERMUX_VERSION" ] || ( [ -n "$PREFIX" ] && [[ "$PREFIX" == *"com.termux"* ]] ); then
echo "[*] Обнаружен Termux — адаптирую .zshrc (замена 'sudo apt install' -> 'apt install')..." echo "[*] Обнаружен Termux — адаптирую managed-конфиг (замена 'sudo apt install' -> 'apt install')..."
if grep -q "sudo apt install" "${ZDOTDIR}/.zshrc" 2>/dev/null; then if grep -q "sudo apt install" "${HOME}/.config/my-zsh/zshrc.managed" 2>/dev/null; then
# делаем inplace замену и удаляем бэкап, если он создастся # делаем inplace замену и удаляем бэкап, если он создастся
sed -i.bak 's/sudo apt install/apt install/g' "${ZDOTDIR}/.zshrc" || true sed -i.bak 's/sudo apt install/apt install/g' "${HOME}/.config/my-zsh/zshrc.managed" || true
rm -f "${ZDOTDIR}/.zshrc.bak" || true rm -f "${HOME}/.config/my-zsh/zshrc.managed.bak" || true
echo "[*] Замена выполнена." echo "[*] Замена выполнена."
else else
echo "[*] В .zshrc не найдено 'sudo apt install' — ничего менять не нужно." echo "[*] В managed-конфиге не найдено 'sudo apt install' — ничего менять не нужно."
fi fi
fi fi
echo "[*] Создаю файл для пользовательских правок: ~/.zshrc.local (если его нет)..."
if [ ! -f "${HOME}/.zshrc.local" ]; then
cat > "${HOME}/.zshrc.local" <<'EOF'
# Your custom zsh config goes here.
# This file will NOT be overwritten by my-zsh updates.
EOF
fi
# ========= Optional configs =========
ask_yes_no() {
# Usage: ask_yes_no "Question" "N|Y"
local question default reply
question="$1"
default="${2:-N}"
# Non-interactive shell: honor default.
if [ ! -t 0 ]; then
[ "${default}" = "Y" ] && return 0 || return 1
fi
read -r -p "${question} [y/N]: " reply
reply="${reply:-$default}"
case "$reply" in
y|Y|yes|YES) return 0 ;;
*) return 1 ;;
esac
}
backup_if_exists() {
local p ts
p="$1"
ts="$(date +%Y%m%d-%H%M%S)"
if [ -f "$p" ]; then
cp -f "$p" "$p.backup.$ts" 2>/dev/null || true
fi
}
# Micro
MICRORC_URL="https://raw.githubusercontent.com/ushst/my-zsh/main/.microrc"
MICRO_SHELL_SYNTAX_URL="https://raw.githubusercontent.com/ushst/my-zsh/main/.config/micro/syntax/shell.yaml"
if [ "${MY_ZSH_INSTALL_MICRO:-}" = "1" ] || ( [ "${MY_ZSH_INSTALL_MICRO:-}" != "0" ] && ask_yes_no "[?] Установить конфиг для micro ( .microrc + syntax )?" "N" ); then
echo "[*] Установка конфига micro..."
backup_if_exists "${HOME}/.microrc"
wget -O "${HOME}/.microrc" "$MICRORC_URL"
mkdir -p "${HOME}/.config/micro/syntax" || true
backup_if_exists "${HOME}/.config/micro/syntax/shell.yaml"
wget -O "${HOME}/.config/micro/syntax/shell.yaml" "$MICRO_SHELL_SYNTAX_URL"
fi
# msfconsole
MSFCONSOLE_RC_URL="https://raw.githubusercontent.com/ushst/my-zsh/main/msfconsole.rc"
if [ "${MY_ZSH_INSTALL_MSFCONSOLE:-}" = "1" ] || ( [ "${MY_ZSH_INSTALL_MSFCONSOLE:-}" != "0" ] && ask_yes_no "[?] Установить конфиг для msfconsole ( alias s -> search )?" "N" ); then
echo "[*] Установка конфига msfconsole..."
mkdir -p "${HOME}/.msf4" || true
backup_if_exists "${HOME}/.msf4/msfconsole.rc"
wget -O "${HOME}/.msf4/msfconsole.rc" "$MSFCONSOLE_RC_URL"
fi
echo "[*] Смена стандартной оболочки на zsh..." echo "[*] Смена стандартной оболочки на zsh..."
chsh -s "$(which zsh)" "$USER" chsh -s "$(which zsh)" "$USER"

View File

@@ -1 +1 @@
1.2.0 1.3.0

76
zshrc.managed Normal file
View File

@@ -0,0 +1,76 @@
# Managed by github.com/ushst/my-zsh (do not edit; it may be overwritten).
# Put your changes into ~/.zshrc.local or ~/.zshrc.d/*.zsh
# If you come from bash you might have to change your $PATH.
# export PATH=$HOME/bin:/usr/local/bin:$PATH
# Path to your oh-my-zsh installation.
export ZSH="$HOME/.oh-my-zsh"
# Theme
ZSH_THEME="avit"
# Plugins
plugins=(
git
zsh-autosuggestions
zsh-syntax-highlighting
)
source "$ZSH/oh-my-zsh.sh"
# User configuration
prompt_context() {
if [[ "$USER" != "$DEFAULT_USER" || -n "$SSH_CLIENT" ]]; then
prompt_segment black default "%(!.%{%F{yellow}%}.)"
fi
}
export NVM_DIR="$HOME/.nvm"
[ -s "$NVM_DIR/nvm.sh" ] && \. "$NVM_DIR/nvm.sh"
[ -s "$NVM_DIR/bash_completion" ] && \. "$NVM_DIR/bash_completion"
# Custom aliases and functions
alias untargz='tar -xvzf'
function a() {
case "$1" in
i)
shift
sudo apt install "$@"
;;
u)
sudo apt update
;;
up)
sudo apt upgrade -y
;;
all)
sudo apt update && sudo apt upgrade -y
;;
*)
echo "Неизвестная команда: a $1"
;;
esac
}
extract() {
if [ -f "$1" ]; then
case "$1" in
*.tar.bz2) tar xvjf "$1" ;;
*.tar.gz) tar xvzf "$1" ;;
*.tar.xz) tar xvJf "$1" ;;
*.tar) tar xvf "$1" ;;
*.gz) gunzip "$1" ;;
*.zip) unzip "$1" ;;
*.rar) unrar x "$1" ;;
*.7z) 7z x "$1" ;;
*) echo "Неизвестный формат: $1" ;;
esac
else
echo "Файл не найден: $1"
fi
}
alias x='extract'