feat: switch to biome, apply formatting, shellcheck (#227)

* feat: switch to biome, apply formatting, shellcheck
* chore: apply cr comments
* chore: few config tweaks, shellcheck hook now py-based
* chore: lint fixes and pr comments
* chore(lint): megalinter, and other fixes

Signed-off-by: Ismo Vuorinen <ismo@ivuorinen.net>
This commit is contained in:
2025-12-17 16:03:29 +02:00
committed by GitHub
parent 4b0e38ffd2
commit 961efec364
69 changed files with 782 additions and 1089 deletions

View File

@@ -9,7 +9,7 @@ insert_final_newline = true
trim_trailing_whitespace = true trim_trailing_whitespace = true
[*.fish] [*.fish]
max_line_length = 80 max_line_length = 120
[*.md] [*.md]
max_line_length = 120 max_line_length = 120
@@ -49,3 +49,6 @@ ignore = true
[plan] [plan]
trim_trailing_whitespace = false trim_trailing_whitespace = false
max_line_length = off max_line_length = off
[base/hammerspoon/hammerspoon.types.lua]
max_line_length = off

View File

@@ -1,3 +0,0 @@
{
"extends": ["@ivuorinen"]
}

View File

@@ -16,10 +16,10 @@ module.exports = {
excludeTypes: [], excludeTypes: [],
renderTypeSection: function (label, commits) { renderTypeSection: (label, commits) => {
let text = `\n## ${label}\n\n` let text = `\n## ${label}\n\n`
commits.forEach(commit => { commits.forEach((commit) => {
const scope = commit.scope ? `**${commit.scope}:** ` : '' const scope = commit.scope ? `**${commit.scope}:** ` : ''
text += `- ${scope}${commit.subject}\n` text += `- ${scope}${commit.subject}\n`
}) })
@@ -27,10 +27,10 @@ module.exports = {
return text return text
}, },
renderChangelog: function (release, changes) { renderChangelog: (release, changes) => {
const now = new Date() const now = new Date()
const d = now.toISOString().substring(0, 10) const d = now.toISOString().substring(0, 10)
const header = `# ${release} - ${d}\n` const header = `# ${release} - ${d}\n`
return header + changes + '\n\n' return `${header}${changes}\n\n`
}, },
} }

View File

@@ -1,19 +0,0 @@
{
"extends": "@ivuorinen/markdownlint-config",
"code-block-style": {
"style": "fenced"
},
"code-fence-style": {
"style": "backtick"
},
"heading-style": {
"style": "atx"
},
"no-duplicate-heading": {
"siblings_only": true
},
"required-headings": false,
"ul-style": {
"style": "dash"
}
}

View File

@@ -16,12 +16,9 @@ SHOW_SKIPPED_LINTERS: false # Show skipped linters in MegaLinter log
TYPESCRIPT_DEFAULT_STYLE: prettier # Default style for TypeScript TYPESCRIPT_DEFAULT_STYLE: prettier # Default style for TypeScript
DISABLE_LINTERS: DISABLE_LINTERS:
- REPOSITORY_DEVSKIM - REPOSITORY_DEVSKIM
- JAVASCRIPT_ES - JAVASCRIPT_ES # using biome
- JAVASCRIPT_PRETTIER # using biome
YAML_YAMLLINT_CONFIG_FILE: .yamllint.yml YAML_YAMLLINT_CONFIG_FILE: .yamllint.yml
MARKDOWN_MARKDOWNLINT_CONFIG_FILE: .markdownlint.json
JAVASCRIPT_ES_CONFIG_FILE: .eslintrc.json
TYPESCRIPT_ES_CONFIG_FILE: .eslintrc.json
REPOSITORY_GIT_DIFF_DISABLE_ERRORS: true REPOSITORY_GIT_DIFF_DISABLE_ERRORS: true
FILTER_REGEX_EXCLUDE: > FILTER_REGEX_EXCLUDE: >
(node_modules|tools|config/cheat/cheatsheets/community|config/cheat/cheatsheets/tldr|config/fzf|config/zsh|config/tmux/plugins) (node_modules|tools|config/cheat/cheatsheets/community|config/cheat/cheatsheets/tldr|config/fzf|config/zsh|config/tmux/plugins)

View File

@@ -20,22 +20,22 @@ repos:
- id: end-of-file-fixer - id: end-of-file-fixer
- id: mixed-line-ending - id: mixed-line-ending
args: [--fix=auto] args: [--fix=auto]
- id: pretty-format-json
args: [--autofix, --no-sort-keys]
- repo: https://github.com/igorshubovych/markdownlint-cli - repo: local
rev: v0.47.0
hooks: hooks:
- id: markdownlint - id: biome-check
args: [-c, .markdownlint.json, --fix] name: Biome Check
entry: yarn biome check --write --files-ignore-unknown=true --no-errors-on-unmatched
language: system
files: \.(js|ts|jsx|tsx|json|md)$
- repo: https://github.com/adrienverge/yamllint - repo: https://github.com/adrienverge/yamllint
rev: v1.37.1 rev: v1.37.1
hooks: hooks:
- id: yamllint - id: yamllint
- repo: https://github.com/koalaman/shellcheck-precommit - repo: https://github.com/shellcheck-py/shellcheck-py
rev: v0.11.0 rev: v0.11.0.1
hooks: hooks:
- id: shellcheck - id: shellcheck
@@ -49,15 +49,11 @@ repos:
hooks: hooks:
- id: actionlint - id: actionlint
- repo: https://github.com/renovatebot/pre-commit-hooks
rev: 42.52.8
hooks:
- id: renovate-config-validator
- repo: https://github.com/JohnnyMorganz/StyLua - repo: https://github.com/JohnnyMorganz/StyLua
rev: v2.3.1 rev: v2.3.1
hooks: hooks:
- id: stylua # or stylua-system / stylua-github - id: stylua # or stylua-system / stylua-github
exclude: hammerspoon\.types\.lua$
- repo: https://github.com/hugoh/pre-commit-fish.git - repo: https://github.com/hugoh/pre-commit-fish.git
rev: v1.2 rev: v1.2

View File

@@ -1,16 +0,0 @@
# vim: ft=gitignore
.mypy_cache/*
Brewfile.lock.json
base/plan
config/cheat/cheatsheets/community
config/cheat/cheatsheets/tldr
config/fzf/*
config/nvim/*
config/op/plugins/used_plugins/*
config/tmux/plugins/*
config/zsh/*
lazy-lock.json
local/bin/antigen.zsh
local/bin/asdf
tools/antidote/*
tools/dotbot*

View File

@@ -1,15 +0,0 @@
module.exports = {
...require('@ivuorinen/prettier-config'),
trailingComma: 'all',
// Add custom options below:
overrides: [
{
files: '*.md',
options: {
printWidth: 120,
proseWrap: 'preserve',
tabWidth: 2,
},
},
],
}

View File

@@ -0,0 +1,5 @@
{
"sonarCloudOrganization": "ivuorinen",
"projectKey": "ivuorinen_dotfiles",
"region": "EU"
}

View File

@@ -193,11 +193,12 @@ end)
-- Paste 1Password secret with Meh + P -- Paste 1Password secret with Meh + P
f18:bind({}, 'p', function() f18:bind({}, 'p', function()
local output, status = hs.execute('op read "op://Svea/3hzhctmvovbwlgulv7mgy25rf4/login-input"', true) local output, status =
hs.execute('op read "op://Svea/3hzhctmvovbwlgulv7mgy25rf4/login-input"', true)
if status then if status then
hs.eventtap.keyStrokes(output:gsub('%s+$', '')) -- trim trailing whitespace hs.eventtap.keyStrokes(output:gsub('%s+$', '')) -- trim trailing whitespace
else else
hs.alert.show('1Password CLI error') hs.alert.show '1Password CLI error'
end end
end) end)

103
biome.json Normal file
View File

@@ -0,0 +1,103 @@
{
"$schema": "https://biomejs.dev/schemas/2.3.1/schema.json",
"vcs": {
"enabled": true,
"clientKind": "git",
"useIgnoreFile": true,
"defaultBranch": "main"
},
"files": {
"ignoreUnknown": true,
"includes": [
"**",
"!!**/.mypy_cache",
"!!**/Brewfile.lock.json",
"!!**/base/plan",
"!!**/config/cheat/cheatsheets/community",
"!!**/config/cheat/cheatsheets/tldr",
"!!**/config/fzf",
"!!**/config/nvim",
"!!**/config/op/plugins/used_plugins",
"!!**/config/tmux/plugins",
"!!**/config/zsh",
"!!**/config/vim",
"!!**/lazy-lock.json",
"!!**/local/bin/antigen.zsh",
"!!**/local/bin/asdf",
"!!**/tools/antidote",
"!!**/tools/dotbot",
"!!**/node_modules"
]
},
"formatter": {
"enabled": true,
"formatWithErrors": false,
"indentStyle": "space",
"indentWidth": 2,
"lineEnding": "lf",
"lineWidth": 80
},
"linter": {
"enabled": true,
"rules": {
"recommended": true,
"correctness": {
"noUnusedVariables": "warn",
"noUnusedImports": "warn"
},
"style": {
"useConst": "warn",
"useTemplate": "warn"
},
"suspicious": {
"noExplicitAny": "warn",
"noConsole": "off"
}
}
},
"javascript": {
"formatter": {
"enabled": true,
"quoteStyle": "single",
"jsxQuoteStyle": "double",
"trailingCommas": "all",
"semicolons": "asNeeded",
"arrowParentheses": "always",
"bracketSpacing": true,
"bracketSameLine": false,
"quoteProperties": "asNeeded",
"indentStyle": "space",
"indentWidth": 2
}
},
"json": {
"parser": {
"allowComments": true,
"allowTrailingCommas": false
},
"formatter": {
"enabled": true,
"indentStyle": "space",
"indentWidth": 2,
"lineWidth": 80
}
},
"overrides": [
{
"includes": ["*.md", "*.mdx"],
"formatter": {
"enabled": true,
"lineWidth": 120
}
},
{
"includes": ["package.json"],
"json": {
"formatter": {
"enabled": true,
"indentWidth": 2
}
}
}
]
}

View File

@@ -6,21 +6,19 @@
* *
* @param {Object} windows - All windows in the current space. * @param {Object} windows - All windows in the current space.
* @param {Object} screenFrame - The frame of the current screen. * @param {Object} screenFrame - The frame of the current screen.
* @param {Object} state - The state of the current space.
* @param {Object} extendedFrames - The frames of the windows in the current space.
* @return {Object} - The frames for the windows in the current space. * @return {Object} - The frames for the windows in the current space.
*/ */
function layout() { function layout() {
return { return {
name: 'Almost Maximize', name: 'Almost Maximize',
getFrameAssignments: (windows, screenFrame, state, extendedFrames) => { getFrameAssignments: (windows, screenFrame) => {
const width = screenFrame.width * 0.95 const width = screenFrame.width * 0.95
const height = screenFrame.height * 0.95 const height = screenFrame.height * 0.95
const x = (screenFrame.width - width) / 2 const x = (screenFrame.width - width) / 2
const y = (screenFrame.height - height) / 2 const y = (screenFrame.height - height) / 2
const windowFrames = {} const windowFrames = {}
windows.forEach(window => { windows.forEach((window) => {
windowFrames[window.id] = { windowFrames[window.id] = {
Y: screenFrame.y + y, Y: screenFrame.y + y,
x: screenFrame.x + x, x: screenFrame.x + x,
@@ -33,3 +31,5 @@ function layout() {
}, },
} }
} }
module.exports = layout()

View File

@@ -1,5 +1,3 @@
#!/usr/bin/env bash
#
# This file is secret and wont be added to the git repo. # This file is secret and wont be added to the git repo.
export GITLAB_API_TOKEN="" export GITLAB_API_TOKEN=""

0
config/fish/completions/docker.fish Executable file → Normal file
View File

View File

@@ -26,7 +26,7 @@ if status is-interactive
# type -q fnm; and fnm env --use-on-cd --shell fish | source # type -q fnm; and fnm env --use-on-cd --shell fish | source
type -q load_nvm; and load_nvm >/dev/stderr type -q load_nvm; and load_nvm >/dev/stderr
# Intialize other tools if available # Initialize other tools if available
type -q zoxide; and zoxide init fish | source type -q zoxide; and zoxide init fish | source
# Start tmux if not already running and not in SSH # Start tmux if not already running and not in SSH
@@ -40,7 +40,7 @@ set -gx PATH $PATH $HOME/.lmstudio/bin
# vim: ft=fish ts=4 sw=4 et: # vim: ft=fish ts=4 sw=4 et:
# opencode # opencode
fish_add_path /Users/ivuorinen/.opencode/bin fish_add_path $HOME/.opencode/bin
# Added by OrbStack: command-line tools and integration # Added by OrbStack: command-line tools and integration
# This won't be added again if you remove it. # This won't be added again if you remove it.

View File

@@ -1,5 +1,3 @@
#!/usr/bin/env fish
# XDG Base Directory Specification # XDG Base Directory Specification
set -q XDG_CONFIG_HOME; or set -x XDG_CONFIG_HOME "$HOME/.config" set -q XDG_CONFIG_HOME; or set -x XDG_CONFIG_HOME "$HOME/.config"
set -q XDG_DATA_HOME; or set -x XDG_DATA_HOME "$HOME/.local/share" set -q XDG_DATA_HOME; or set -x XDG_DATA_HOME "$HOME/.local/share"

View File

@@ -26,7 +26,6 @@ Options:
return 1 return 1
end end
if set --query _flag_exact if set --query _flag_exact
set --function apps (__macos_app_find --exact $argv) set --function apps (__macos_app_find --exact $argv)
or return 1 or return 1

View File

@@ -139,7 +139,6 @@ Options:
return 0 return 0
end end
__macos_mac_touchid_sudo::check_supported __macos_mac_touchid_sudo::check_supported
or return or return

View File

@@ -8,7 +8,9 @@ function __ssh_agent_is_started -d "check if ssh agent is already started"
end end
end end
if begin; test -f "$SSH_ENV"; and test -z "$SSH_AGENT_PID"; end if begin
test -f "$SSH_ENV"; and test -z "$SSH_AGENT_PID"
end
source $SSH_ENV >/dev/null source $SSH_ENV >/dev/null
end end

View File

@@ -3,10 +3,8 @@ function _fzf_extract_var_info --argument-names variable_name set_show_output --
# Extract only the lines about the variable, all of which begin with either # Extract only the lines about the variable, all of which begin with either
# $variable_name: ...or... $variable_name[ # $variable_name: ...or... $variable_name[
string match --regex "^\\\$$variable_name(?::|\[).*" <$set_show_output | string match --regex "^\\\$$variable_name(?::|\[).*" <$set_show_output |
# Strip the variable name prefix, including ": " for scope info lines # Strip the variable name prefix, including ": " for scope info lines
string replace --regex "^\\\$$variable_name(?:: )?" '' | string replace --regex "^\\\$$variable_name(?:: )?" '' |
# Distill the lines of values, replacing... # Distill the lines of values, replacing...
# [1]: |value| # [1]: |value|
# ...with... # ...with...

View File

@@ -24,7 +24,6 @@ function _fzf_search_directory --description "Search the current directory. Repl
set -f file_paths_selected ($fd_cmd 2>/dev/null | _fzf_wrapper $fzf_arguments) set -f file_paths_selected ($fd_cmd 2>/dev/null | _fzf_wrapper $fzf_arguments)
end end
if test $status -eq 0 if test $status -eq 0
commandline --current-token --replace -- (string escape -- $file_paths_selected | string join ' ') commandline --current-token --replace -- (string escape -- $file_paths_selected | string join ' ')
end end

View File

@@ -1,7 +1,7 @@
function bass function bass
set -l bash_args $argv set -l bash_args $argv
set -l bass_debug set -l bass_debug
if test "$bash_args[1]_" = '-d_' if test "$bash_args[1]_" = -d_
set bass_debug true set bass_debug true
set -e bash_args[1] set -e bash_args[1]
end end

View File

@@ -4,7 +4,7 @@ function load_nvm --on-variable="PWD"
set -l nvmrc_path (nvm_find_nvmrc) set -l nvmrc_path (nvm_find_nvmrc)
if test -n "$nvmrc_path" if test -n "$nvmrc_path"
set -l nvmrc_node_version (nvm version (cat $nvmrc_path)) set -l nvmrc_node_version (nvm version (cat $nvmrc_path))
if test "$nvmrc_node_version" = "N/A" if test "$nvmrc_node_version" = N/A
nvm install (cat $nvmrc_path) nvm install (cat $nvmrc_path)
else if test "$nvmrc_node_version" != "$node_version" else if test "$nvmrc_node_version" != "$node_version"
nvm use $nvmrc_node_version nvm use $nvmrc_node_version

View File

@@ -158,7 +158,7 @@ function paths --description "Reveal the executable matches in shell paths or fi
# check # check
set -l built (type --type $input 12&>/dev/null) set -l built (type --type $input 12&>/dev/null)
if test -n "$built" if test -n "$built"
and test "$built" = 'builtin' and test "$built" = builtin
set $foundStatus 0 set $foundStatus 0
if not set -q _flag_c if not set -q _flag_c
echo -e -n "builtin\n" echo -e -n "builtin\n"

1
config/fzf/completion.bash Executable file → Normal file
View File

@@ -1,3 +1,4 @@
# shellcheck disable=all
# ____ ____ # ____ ____
# / __/___ / __/ # / __/___ / __/
# / /_/_ / / /_ # / /_/_ / / /_

1
config/fzf/completion.zsh Executable file → Normal file
View File

@@ -1,3 +1,4 @@
# shellcheck disable=all
# ____ ____ # ____ ____
# / __/___ / __/ # / __/___ / __/
# / /_/_ / / /_ # / /_/_ / / /_

View File

@@ -1,3 +1,4 @@
# shellcheck shell=bash
# Everforest theme for fzf # Everforest theme for fzf
# Generated from template - do not edit manually # Generated from template - do not edit manually

0
config/fzf/fzf.bash Executable file → Normal file
View File

View File

@@ -1,3 +1,5 @@
#!/usr/bin/env zsh
# shellcheck disable=SC1071
# Setup fzf # Setup fzf
# --------- # ---------

9
config/fzf/key-bindings.bash Executable file → Normal file
View File

@@ -20,8 +20,8 @@ __fzf_select__() {
-o -type d -print \ -o -type d -print \
-o -type l -print 2> /dev/null | cut -b3-"}" -o -type l -print 2> /dev/null | cut -b3-"}"
opts="--height ${FZF_TMUX_HEIGHT:-40%} --bind=ctrl-z:ignore --reverse ${FZF_DEFAULT_OPTS-} ${FZF_CTRL_T_OPTS-} -m" opts="--height ${FZF_TMUX_HEIGHT:-40%} --bind=ctrl-z:ignore --reverse ${FZF_DEFAULT_OPTS-} ${FZF_CTRL_T_OPTS-} -m"
eval "$cmd" | # shellcheck disable=SC2091 # Intentionally execute output of __fzfcmd
FZF_DEFAULT_OPTS="$opts" $(__fzfcmd) "$@" | eval "$cmd" | FZF_DEFAULT_OPTS="$opts" $(__fzfcmd) "$@" |
while read -r item; do while read -r item; do
printf '%q ' "$item" # escape special chars printf '%q ' "$item" # escape special chars
done done
@@ -35,7 +35,8 @@ if [[ $- =~ i ]]; then
} }
fzf-file-widget() { fzf-file-widget() {
local selected="$(__fzf_select__ "$@")" local selected
selected="$(__fzf_select__ "$@")"
READLINE_LINE="${READLINE_LINE:0:$READLINE_POINT}$selected${READLINE_LINE:$READLINE_POINT}" READLINE_LINE="${READLINE_LINE:0:$READLINE_POINT}$selected${READLINE_LINE:$READLINE_POINT}"
READLINE_POINT=$((READLINE_POINT + ${#selected})) READLINE_POINT=$((READLINE_POINT + ${#selected}))
} }
@@ -45,6 +46,7 @@ if [[ $- =~ i ]]; then
cmd="${FZF_ALT_C_COMMAND:-"command find -L . -mindepth 1 \\( -path '*/\\.*' -o -fstype 'sysfs' -o -fstype 'devfs' -o -fstype 'devtmpfs' -o -fstype 'proc' \\) -prune \ cmd="${FZF_ALT_C_COMMAND:-"command find -L . -mindepth 1 \\( -path '*/\\.*' -o -fstype 'sysfs' -o -fstype 'devfs' -o -fstype 'devtmpfs' -o -fstype 'proc' \\) -prune \
-o -type d -print 2> /dev/null | cut -b3-"}" -o -type d -print 2> /dev/null | cut -b3-"}"
opts="--height ${FZF_TMUX_HEIGHT:-40%} --bind=ctrl-z:ignore --reverse ${FZF_DEFAULT_OPTS-} ${FZF_ALT_C_OPTS-} +m" opts="--height ${FZF_TMUX_HEIGHT:-40%} --bind=ctrl-z:ignore --reverse ${FZF_DEFAULT_OPTS-} ${FZF_ALT_C_OPTS-} +m"
# shellcheck disable=SC2091 # Intentionally execute output of __fzfcmd
dir=$( dir=$(
set +o pipefail set +o pipefail
eval "$cmd" | FZF_DEFAULT_OPTS="$opts" $(__fzfcmd) eval "$cmd" | FZF_DEFAULT_OPTS="$opts" $(__fzfcmd)
@@ -55,6 +57,7 @@ if [[ $- =~ i ]]; then
local output opts script local output opts script
opts="--height ${FZF_TMUX_HEIGHT:-40%} --bind=ctrl-z:ignore ${FZF_DEFAULT_OPTS-} -n2..,.. --scheme=history --bind=ctrl-r:toggle-sort ${FZF_CTRL_R_OPTS-} +m --read0" opts="--height ${FZF_TMUX_HEIGHT:-40%} --bind=ctrl-z:ignore ${FZF_DEFAULT_OPTS-} -n2..,.. --scheme=history --bind=ctrl-r:toggle-sort ${FZF_CTRL_R_OPTS-} +m --read0"
script='BEGIN { getc; $/ = "\n\t"; $HISTCOUNT = $ENV{last_hist} + 1 } s/^[ *]//; print $HISTCOUNT - $. . "\t$_" if !$seen{$_}++' script='BEGIN { getc; $/ = "\n\t"; $HISTCOUNT = $ENV{last_hist} + 1 } s/^[ *]//; print $HISTCOUNT - $. . "\t$_" if !$seen{$_}++'
# shellcheck disable=SC2091 # Intentionally execute output of __fzfcmd
output=$( output=$(
set +o pipefail set +o pipefail
builtin fc -lnr -2147483648 | builtin fc -lnr -2147483648 |

9
config/fzf/key-bindings.fish Executable file → Normal file
View File

@@ -33,7 +33,9 @@ function fzf_key_bindings
test -n "$FZF_TMUX_HEIGHT"; or set FZF_TMUX_HEIGHT 40% test -n "$FZF_TMUX_HEIGHT"; or set FZF_TMUX_HEIGHT 40%
begin begin
set -lx FZF_DEFAULT_OPTS "--height $FZF_TMUX_HEIGHT --reverse --bind=ctrl-z:ignore $FZF_DEFAULT_OPTS $FZF_CTRL_T_OPTS" set -lx FZF_DEFAULT_OPTS "--height $FZF_TMUX_HEIGHT --reverse --bind=ctrl-z:ignore $FZF_DEFAULT_OPTS $FZF_CTRL_T_OPTS"
eval "$FZF_CTRL_T_COMMAND | "(__fzfcmd)' -m --query "'$fzf_query'"' | while read -l r; set result $result $r; end eval "$FZF_CTRL_T_COMMAND | "(__fzfcmd)' -m --query "'$fzf_query'"' | while read -l r
set result $result $r
end
end end
if [ -z "$result" ] if [ -z "$result" ]
commandline -f repaint commandline -f repaint
@@ -61,7 +63,8 @@ function fzf_key_bindings
# history's -z flag is needed for multi-line support. # history's -z flag is needed for multi-line support.
# history's -z flag was added in fish 2.4.0, so don't use it for versions # history's -z flag was added in fish 2.4.0, so don't use it for versions
# before 2.4.0. # before 2.4.0.
if [ "$FISH_MAJOR" -gt 2 -o \( "$FISH_MAJOR" -eq 2 -a "$FISH_MINOR" -ge 4 \) ]; if [ "$FISH_MAJOR" -gt 2 -o \( "$FISH_MAJOR" -eq 2 -a "$FISH_MINOR" -ge 4 \) ]
history -z | eval (__fzfcmd) --read0 --print0 -q '(commandline)' | read -lz result history -z | eval (__fzfcmd) --read0 --print0 -q '(commandline)' | read -lz result
and commandline -- $result and commandline -- $result
else else
@@ -106,7 +109,7 @@ function fzf_key_bindings
else if [ $FZF_TMUX -eq 1 ] else if [ $FZF_TMUX -eq 1 ]
echo "fzf-tmux -d$FZF_TMUX_HEIGHT -- " echo "fzf-tmux -d$FZF_TMUX_HEIGHT -- "
else else
echo "fzf" echo fzf
end end
end end

1
config/fzf/key-bindings.zsh Executable file → Normal file
View File

@@ -1,3 +1,4 @@
# shellcheck disable=all
# ____ ____ # ____ ____
# / __/___ / __/ # / __/___ / __/
# / /_/_ / / /_ # / /_/_ / / /_

View File

@@ -55,7 +55,8 @@ return {
{ {
event = 'file_opened', event = 'file_opened',
handler = function(_) handler = function(_)
require('neo-tree.command').execute { action = 'close' } local c = require 'neo-tree.command'
c.execute { action = 'close' }
end, end,
}, },
}, },

View File

@@ -56,4 +56,3 @@ disable_startup_command = true
name = "Downloads" name = "Downloads"
path = "~/Downloads" path = "~/Downloads"
startup_command = "lsa" startup_command = "lsa"

View File

@@ -1,2 +1 @@
set -g @catppuccin_flavor "mocha" set -g @catppuccin_flavor "mocha"

View File

@@ -1,4 +1,4 @@
# shellcheck shell=zsh # shellcheck disable=SC1071,SC1103,SC2148
# Source: https://github.com/nvm-sh/nvm#zsh # Source: https://github.com/nvm-sh/nvm#zsh
# place this after nvm initialization! # place this after nvm initialization!
autoload -U add-zsh-hook autoload -U add-zsh-hook

52
cspell.config.yaml Normal file
View File

@@ -0,0 +1,52 @@
version: "0.2"
ignorePaths: []
dictionaryDefinitions: []
dictionaries: []
words:
- aquasec
- cdgr
- cheatsheets
- CODEQUALITY
- DEVSKIM
- dscacheutil
- emptytrash
- flushcache
- getifaddr
- goenv
- irssi
- Ismo
- isodate
- KEYSFILE
- KEYSSOURCE
- killall
- lakka
- libexec
- Licence
- lmstudio
- localip
- locatedb
- LOGFILE
- luarocks
- mvdan
- myip
- onnimonni
- opencode
- optstring
- orbstack
- osascript
- phpenv
- psub
- pyenv
- SARIF
- shellcheck
- shfmt
- shivammathur
- spelllang
- updatedb
- vendordirs
- vimrc
- virtualenv
- Vuorinen
- zedit
ignoreWords: []
import: []

View File

@@ -87,4 +87,3 @@ Leader: `<ctrl><space>`
S-Left Move the visible part of the window left S-Left Move the visible part of the window left
S-Right Move the visible part of the window right S-Right Move the visible part of the window right
``` ```

View File

@@ -103,9 +103,9 @@ fetch_keys_if_missing()
if [[ ! -f "$KEYS_FILE" ]]; then if [[ ! -f "$KEYS_FILE" ]]; then
log_message "Keys file '$KEYS_FILE' not found. Attempting to fetch from $KEYS_SOURCE..." log_message "Keys file '$KEYS_FILE' not found. Attempting to fetch from $KEYS_SOURCE..."
mkdir -p "$(dirname "$KEYS_FILE")" mkdir -p "$(dirname "$KEYS_FILE")"
curl -s "$KEYS_SOURCE" -o "$KEYS_FILE"
if [[ $? -ne 0 || ! -s "$KEYS_FILE" ]]; then if ! curl -s "$KEYS_SOURCE" -o "$KEYS_FILE" || [[ ! -s "$KEYS_FILE" ]]; then
rm -f "$KEYS_FILE" 2> /dev/null || true
log_message "Error: Failed to fetch keys from $KEYS_SOURCE" log_message "Error: Failed to fetch keys from $KEYS_SOURCE"
exit 1 exit 1
fi fi
@@ -126,10 +126,12 @@ encrypt_file_or_directory()
elif [[ -f "$file" ]]; then elif [[ -f "$file" ]]; then
fetch_keys_if_missing fetch_keys_if_missing
local output_file="${file}.age" local output_file="${file}.age"
age -R "$KEYS_FILE" "$file" > "$output_file" local temp_file
if [[ $? -eq 0 ]]; then temp_file="$(mktemp -p "$(dirname "$file")")"
if age -R "$KEYS_FILE" "$file" > "$temp_file" && mv "$temp_file" "$output_file"; then
log_message "File encrypted successfully: $output_file" log_message "File encrypted successfully: $output_file"
else else
rm -f "$temp_file"
log_message "Error: Failed to encrypt file '$file'." log_message "Error: Failed to encrypt file '$file'."
exit 1 exit 1
fi fi
@@ -147,10 +149,12 @@ decrypt_file_or_directory()
elif [[ -f "$file" ]]; then elif [[ -f "$file" ]]; then
fetch_keys_if_missing fetch_keys_if_missing
local output_file="${file%.age}" local output_file="${file%.age}"
age -d -i "$KEYS_FILE" "$file" > "$output_file" local temp_file
if [[ $? -eq 0 ]]; then temp_file="$(mktemp -p "$(dirname "$file")")"
if age -d -i "$KEYS_FILE" "$file" > "$temp_file" && mv "$temp_file" "$output_file"; then
log_message "File decrypted successfully: $output_file" log_message "File decrypted successfully: $output_file"
else else
rm -f "$temp_file"
log_message "Error: Failed to decrypt file '$file'." log_message "Error: Failed to decrypt file '$file'."
exit 1 exit 1
fi fi

View File

@@ -36,9 +36,8 @@ if [[ ! -f "$KEYS_FILE" ]]; then
mkdir -p "$(dirname "$KEYS_FILE")" mkdir -p "$(dirname "$KEYS_FILE")"
# Fetch the keys and save to the file # Fetch the keys and save to the file
curl -s "$KEYS_SOURCE" -o "$KEYS_FILE" if ! curl -s "$KEYS_SOURCE" -o "$KEYS_FILE" || [[ ! -s "$KEYS_FILE" ]]; then
rm -f "$KEYS_FILE" 2> /dev/null || true
if [[ $? -ne 0 || ! -s "$KEYS_FILE" ]]; then
echo "Error: Failed to fetch keys from $KEYS_SOURCE" echo "Error: Failed to fetch keys from $KEYS_SOURCE"
exit 1 exit 1
fi fi
@@ -50,11 +49,11 @@ fi
# Decrypt the file # Decrypt the file
OUTPUT_FILE="${FILE%.age}" OUTPUT_FILE="${FILE%.age}"
age -d -i "$KEYS_FILE" "$FILE" > "$OUTPUT_FILE" TEMP_FILE="$(mktemp -p "$(dirname "$OUTPUT_FILE")")"
if age -d -i "$KEYS_FILE" "$FILE" > "$TEMP_FILE" && mv "$TEMP_FILE" "$OUTPUT_FILE"; then
if [[ $? -eq 0 ]]; then
echo "File decrypted successfully: $OUTPUT_FILE" echo "File decrypted successfully: $OUTPUT_FILE"
else else
rm -f "$TEMP_FILE"
echo "Error: Failed to decrypt file." echo "Error: Failed to decrypt file."
exit 1 exit 1
fi fi

View File

@@ -37,9 +37,7 @@ if [[ ! -f "$KEYS_FILE" ]]; then
mkdir -p "$(dirname "$KEYS_FILE")" mkdir -p "$(dirname "$KEYS_FILE")"
# Fetch the keys and save to the file # Fetch the keys and save to the file
curl -s "$KEYS_SOURCE" -o "$KEYS_FILE" if ! curl -s "$KEYS_SOURCE" -o "$KEYS_FILE" || [[ ! -s "$KEYS_FILE" ]]; then
if [[ $? -ne 0 || ! -s "$KEYS_FILE" ]]; then
echo "Error: Failed to fetch keys from $KEYS_SOURCE" echo "Error: Failed to fetch keys from $KEYS_SOURCE"
exit 1 exit 1
fi fi
@@ -51,9 +49,7 @@ fi
# Encrypt the file # Encrypt the file
OUTPUT_FILE="${FILE}.age" OUTPUT_FILE="${FILE}.age"
age -R "$KEYS_FILE" "$FILE" > "$OUTPUT_FILE" if age -R "$KEYS_FILE" "$FILE" > "$OUTPUT_FILE"; then
if [[ $? -eq 0 ]]; then
echo "File encrypted successfully: $OUTPUT_FILE" echo "File encrypted successfully: $OUTPUT_FILE"
else else
echo "Error: Failed to encrypt file." echo "Error: Failed to encrypt file."

View File

@@ -25,6 +25,7 @@ source_file()
case "$CURRENT_SHELL" in case "$CURRENT_SHELL" in
fish) fish)
if [[ -f "$file.fish" ]]; then if [[ -f "$file.fish" ]]; then
# shellcheck disable=SC1090
source "$file.fish" source "$file.fish"
else else
echo "Fish shell file not found: $file.fish" echo "Fish shell file not found: $file.fish"
@@ -32,6 +33,7 @@ source_file()
fi fi
;; ;;
sh | bash | zsh) sh | bash | zsh)
# shellcheck disable=SC1090
source "$file" source "$file"
;; ;;
*) *)

View File

@@ -142,7 +142,7 @@ if [[ ! "$opt" =~ "-E" ]] && tmux list-panes -F '#F' | grep -q Z; then
zoomed_without_popup=1 zoomed_without_popup=1
original_window=$(tmux display-message -p "#{window_id}") original_window=$(tmux display-message -p "#{window_id}")
tmp_window=$(tmux new-window -d -P -F "#{window_id}" "bash -c 'while :; do for c in \\| / - '\\;' do sleep 0.2; printf \"\\r\$c fzf-tmux is running\\r\"; done; done'") tmp_window=$(tmux new-window -d -P -F "#{window_id}" "bash -c 'while :; do for c in \\| / - '\\;' do sleep 0.2; printf \"\\r\$c fzf-tmux is running\\r\"; done; done'")
tmux swap-pane -t $tmp_window \; select-window -t $tmp_window tmux swap-pane -t "$tmp_window" \; select-window -t "$tmp_window"
fi fi
set -e set -e
@@ -154,15 +154,16 @@ fifo1="${TMPDIR:-/tmp}/fzf-fifo1-$id"
fifo2="${TMPDIR:-/tmp}/fzf-fifo2-$id" fifo2="${TMPDIR:-/tmp}/fzf-fifo2-$id"
fifo3="${TMPDIR:-/tmp}/fzf-fifo3-$id" fifo3="${TMPDIR:-/tmp}/fzf-fifo3-$id"
if tmux_win_opts=$(tmux show-options -p remain-on-exit \; show-options -p synchronize-panes 2> /dev/null); then if tmux_win_opts=$(tmux show-options -p remain-on-exit \; show-options -p synchronize-panes 2> /dev/null); then
tmux_win_opts=($(sed '/ off/d; s/synchronize-panes/set-option -p synchronize-panes/; s/remain-on-exit/set-option -p remain-on-exit/; s/$/ \\;/' <<< "$tmux_win_opts")) mapfile -t tmux_win_opts < <(sed '/ off/d; s/synchronize-panes/set-option -p synchronize-panes/; s/remain-on-exit/set-option -p remain-on-exit/; s/$/ \\;/' <<< "$tmux_win_opts")
tmux_off_opts='; set-option -p synchronize-panes off ; set-option -p remain-on-exit off' tmux_off_opts='; set-option -p synchronize-panes off ; set-option -p remain-on-exit off'
else else
tmux_win_opts=($(tmux show-window-options remain-on-exit \; show-window-options synchronize-panes | sed '/ off/d; s/^/set-window-option /; s/$/ \\;/')) mapfile -t tmux_win_opts < <(tmux show-window-options remain-on-exit \; show-window-options synchronize-panes | sed '/ off/d; s/^/set-window-option /; s/$/ \\;/')
tmux_off_opts='; set-window-option synchronize-panes off ; set-window-option remain-on-exit off' tmux_off_opts='; set-window-option synchronize-panes off ; set-window-option remain-on-exit off'
fi fi
# shellcheck disable=SC2329
cleanup() cleanup()
{ {
\rm -f $argsf $fifo1 $fifo2 $fifo3 \rm -f "$argsf" "$fifo1" "$fifo2" "$fifo3"
# Restore tmux window options # Restore tmux window options
if [[ "${#tmux_win_opts[@]}" -gt 0 ]]; then if [[ "${#tmux_win_opts[@]}" -gt 0 ]]; then
@@ -172,9 +173,9 @@ cleanup()
# Remove temp window if we were zoomed without popup options # Remove temp window if we were zoomed without popup options
if [[ -n "$zoomed_without_popup" ]]; then if [[ -n "$zoomed_without_popup" ]]; then
tmux display-message -p "#{window_id}" > /dev/null tmux display-message -p "#{window_id}" > /dev/null
tmux swap-pane -t $original_window \; \ tmux swap-pane -t "$original_window" \; \
select-window -t $original_window \; \ select-window -t "$original_window" \; \
kill-window -t $tmp_window \; \ kill-window -t "$tmp_window" \; \
resize-pane -Z resize-pane -Z
fi fi
@@ -209,39 +210,40 @@ echo "$envs;" > "$argsf"
opts=$(printf "%q " "${args[@]}") opts=$(printf "%q " "${args[@]}")
pppid=$$ pppid=$$
echo -n "trap 'kill -SIGUSR1 -$pppid' EXIT SIGINT SIGTERM;" >> $argsf echo -n "trap 'kill -SIGUSR1 -$pppid' EXIT SIGINT SIGTERM;" >> "$argsf"
close="; trap - EXIT SIGINT SIGTERM $close" close="; trap - EXIT SIGINT SIGTERM $close"
export TMUX=$(cut -d , -f 1,2 <<< "$TMUX") export TMUX
mkfifo -m o+w $fifo2 TMUX=$(cut -d , -f 1,2 <<< "$TMUX")
mkfifo -m o+w "$fifo2"
if [[ "$opt" =~ "-E" ]]; then if [[ "$opt" =~ "-E" ]]; then
cat $fifo2 & cat "$fifo2" &
if [[ -n "$term" ]] || [[ -t 0 ]]; then if [[ -n "$term" ]] || [[ -t 0 ]]; then
cat <<< "\"$fzf\" $opts > $fifo2; out=\$? $close; exit \$out" >> $argsf cat <<< "\"$fzf\" $opts > $fifo2; out=\$? $close; exit \$out" >> "$argsf"
else else
mkfifo $fifo1 mkfifo "$fifo1"
cat <<< "\"$fzf\" $opts < $fifo1 > $fifo2; out=\$? $close; exit \$out" >> $argsf cat <<< "\"$fzf\" $opts < $fifo1 > $fifo2; out=\$? $close; exit \$out" >> "$argsf"
cat <&0 > $fifo1 & cat <&0 > "$fifo1" &
fi fi
tmux popup -d "$PWD" $opt "bash $argsf" > /dev/null 2>&1 tmux popup -d "$PWD" "$opt" "bash $argsf" > /dev/null 2>&1
exit $? exit $?
fi fi
mkfifo -m o+w $fifo3 mkfifo -m o+w "$fifo3"
if [[ -n "$term" ]] || [[ -t 0 ]]; then if [[ -n "$term" ]] || [[ -t 0 ]]; then
cat <<< "\"$fzf\" $opts > $fifo2; echo \$? > $fifo3 $close" >> $argsf cat <<< "\"$fzf\" $opts > $fifo2; echo \$? > $fifo3 $close" >> "$argsf"
else else
mkfifo $fifo1 mkfifo "$fifo1"
cat <<< "\"$fzf\" $opts < $fifo1 > $fifo2; echo \$? > $fifo3 $close" >> $argsf cat <<< "\"$fzf\" $opts < $fifo1 > $fifo2; echo \$? > $fifo3 $close" >> "$argsf"
cat <&0 > $fifo1 & cat <&0 > "$fifo1" &
fi fi
tmux \ tmux \
split-window -c "$PWD" $opt "bash -c 'exec -a fzf bash $argsf'" $swap \ split-window -c "$PWD" "$opt" "bash -c 'exec -a fzf bash $argsf'" "$swap" \
$tmux_off_opts \ "$tmux_off_opts" \
> /dev/null 2>&1 || { > /dev/null 2>&1 || {
"$fzf" "${args[@]}" "$fzf" "${args[@]}"
exit $? exit $?
} }
cat $fifo2 cat "$fifo2"
exit "$(cat $fifo3)" exit "$(cat "$fifo3")"

View File

@@ -25,27 +25,27 @@ fi
# Output functions # Output functions
msg_err() msg_err()
{ {
echo -e "\e[31m$@\e[0m" >&2 echo -e "\e[31m$*\e[0m" >&2
} }
msg_success() msg_success()
{ {
echo -e "\e[32m$@\e[0m" echo -e "\e[32m$*\e[0m"
} }
msg_warn() msg_warn()
{ {
echo -e "\e[33m$@\e[0m" >&2 echo -e "\e[33m$*\e[0m" >&2
} }
msg_info() msg_info()
{ {
echo -e "\e[36m$@\e[0m" echo -e "\e[36m$*\e[0m"
} }
msg_debug() msg_debug()
{ {
[[ $VERBOSE -eq 1 ]] && echo -e "\e[35m$@\e[0m" [[ $VERBOSE -eq 1 ]] && echo -e "\e[35m$*\e[0m"
} }
show_help() show_help()
@@ -180,7 +180,8 @@ format_rule()
get_file_extension() get_file_extension()
{ {
local file="$1" local file="$1"
local basename=$(basename "$file") local basename
basename=$(basename "$file")
local extension="" local extension=""
# Check if file has no extension or is a dotfile # Check if file has no extension or is a dotfile
@@ -231,8 +232,8 @@ suggest_rule()
# If file path contains .d/ pattern, we need special handling # If file path contains .d/ pattern, we need special handling
if [[ "$file" =~ \.d/ ]]; then if [[ "$file" =~ \.d/ ]]; then
# Extract the pattern part that includes the .d/ directory # Extract the pattern part that includes the .d/ directory
local dir_part=$(dirname "$file") local dir_part
local base_name=$(basename "$file") dir_part=$(dirname "$file")
# Check if it's a config directory pattern worth capturing # Check if it's a config directory pattern worth capturing
if [[ "$dir_part" =~ /(\.d|[^/]+\.d)$ ]]; then if [[ "$dir_part" =~ /(\.d|[^/]+\.d)$ ]]; then
@@ -430,7 +431,8 @@ detect_shell_scripts()
fi fi
# Skip if file extension already covered # Skip if file extension already covered
local extension=$(get_file_extension "$rel_path") local extension
extension=$(get_file_extension "$rel_path")
if [[ "$extension" != "$rel_path" ]] && grep -q "^\*\.$extension " <<< "$existing_rules"; then if [[ "$extension" != "$rel_path" ]] && grep -q "^\*\.$extension " <<< "$existing_rules"; then
msg_debug "Script covered by extension rule: $rel_path (*.$extension)" msg_debug "Script covered by extension rule: $rel_path (*.$extension)"
continue continue
@@ -458,7 +460,8 @@ detect_shell_scripts()
fi fi
# Group by directory # Group by directory
local dir=$(dirname "$rel_path") local dir
dir=$(dirname "$rel_path")
if [[ "$dir" == "." ]]; then if [[ "$dir" == "." ]]; then
dir="root" dir="root"
fi fi
@@ -480,7 +483,8 @@ detect_shell_scripts()
# Check if we can use directory-based rules instead of individual files # Check if we can use directory-based rules instead of individual files
for dir in "${!scripts_by_dir[@]}"; do for dir in "${!scripts_by_dir[@]}"; do
local files_in_dir=$(echo -e "${scripts_by_dir[$dir]}" | wc -l) local files_in_dir
files_in_dir=$(echo -e "${scripts_by_dir[$dir]}" | wc -l)
local dir_path="$dir" local dir_path="$dir"
if [[ "$dir" == "root" ]]; then if [[ "$dir" == "root" ]]; then
@@ -575,7 +579,8 @@ suggest_gitattributes()
declare -A seen_patterns=() declare -A seen_patterns=()
while IFS= read -r file; do while IFS= read -r file; do
local suggestion=$(suggest_rule "$file") local suggestion
suggestion=$(suggest_rule "$file")
if [[ -n "$suggestion" ]]; then if [[ -n "$suggestion" ]]; then
IFS=':' read -r pattern attributes <<< "$suggestion" IFS=':' read -r pattern attributes <<< "$suggestion"
@@ -639,7 +644,8 @@ suggest_gitattributes()
formatted_suggestions+="$pattern\n" formatted_suggestions+="$pattern\n"
echo "$pattern" echo "$pattern"
else else
local formatted_rule=$(printf "%-${format_width}s %s\n" "$pattern" "$attributes") local formatted_rule
formatted_rule=$(printf "%-${format_width}s %s\n" "$pattern" "$attributes")
formatted_suggestions+="$formatted_rule\n" formatted_suggestions+="$formatted_rule\n"
echo "$formatted_rule" echo "$formatted_rule"
fi fi

1
local/bin/iterm2_shell_integration.zsh Executable file → Normal file
View File

@@ -1,3 +1,4 @@
# shellcheck disable=all
# This program is free software; you can redistribute it and/or # This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License # modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2 # as published by the Free Software Foundation; either version 2

View File

@@ -142,7 +142,7 @@ if [ ! -x "${CURL}" ]; then
exit 1 exit 1
fi fi
devices="$(printf "${devices}" | xargs -n1 | sort -u | uniq)\n" devices="$(printf "%s" "${devices}" | xargs -n1 | sort -u)"
if [ -z "${devices}" ]; then if [ -z "${devices}" ]; then
__pushover_send_message __pushover_send_message

View File

@@ -126,11 +126,11 @@ if [[ -f "$CONFIG_FILE" ]]; then
PROTECTED_KEYS+=("$key") PROTECTED_KEYS+=("$key")
done <<< "$CUSTOM_KEYS" done <<< "$CUSTOM_KEYS"
SKIPPED+=("$(yq '.skipped_keys[]' "$CONFIG_FILE")") mapfile -t SKIPPED < <(yq '.skipped_keys[]' "$CONFIG_FILE")
while IFS= read -r key; do for key in "${SKIPPED[@]}"; do
# Add to default_skipped_keys # Add to default_skipped_keys
SKIPPED_KEYS+=("$key") SKIPPED_KEYS+=("$key")
done <<< "$SKIPPED" done
CUSTOM_GROUPS=$(yq '.custom_grouping[]' "$CONFIG_FILE") CUSTOM_GROUPS=$(yq '.custom_grouping[]' "$CONFIG_FILE")
while IFS= read -r group; do while IFS= read -r group; do
@@ -159,6 +159,7 @@ is_protected()
return 0 return 0
fi fi
# Wildcard match (protected_key contains '*') # Wildcard match (protected_key contains '*')
# shellcheck disable=SC2053 # Intentional glob matching - protected_key contains wildcard patterns
if [[ "$protected_key" == *"*"* ]] && [[ "$key" == $protected_key ]]; then if [[ "$protected_key" == *"*"* ]] && [[ "$key" == $protected_key ]]; then
return 0 return 0
fi fi
@@ -198,6 +199,7 @@ is_skipped()
return 0 return 0
fi fi
# Wildcard match (skipped_key contains '*') # Wildcard match (skipped_key contains '*')
# shellcheck disable=SC2053 # Intentional glob matching - skipped_key contains wildcard patterns
if [[ "$skipped_key" == *"*"* ]] && [[ "$key" == $skipped_key ]]; then if [[ "$skipped_key" == *"*"* ]] && [[ "$key" == $skipped_key ]]; then
return 0 return 0
fi fi
@@ -244,8 +246,15 @@ while IFS='=' read -r key value; do
value="[protected value]" value="[protected value]"
fi fi
# Update group data # Update group data - check if group already exists
if [[ ! " ${all_groups[*]} " =~ " $group " ]]; then group_exists=false
for existing_group in "${all_groups[@]}"; do
if [[ "$existing_group" == "$group" ]]; then
group_exists=true
break
fi
done
if [[ "$group_exists" == false ]]; then
all_groups+=("$group") all_groups+=("$group")
fi fi

View File

@@ -30,10 +30,9 @@ config_msg()
{ {
# if $1 is empty, return # if $1 is empty, return
[ -z "$1" ] && return [ -z "$1" ] && return
[ -z "$2" ] && $2=""
local msg_type="$1" local msg_type="$1"
local msg_content="$2" local msg_content="${2:-}"
[[ "$VERBOSE" -eq 1 ]] && printf 'x-load-configs: %s %s\n' "$msg_type" "$msg_content" [[ "$VERBOSE" -eq 1 ]] && printf 'x-load-configs: %s %s\n' "$msg_type" "$msg_content"
return 0 return 0
} }
@@ -50,7 +49,8 @@ source_config()
{ {
local config_file=$1 local config_file=$1
if [ -f "$config_file" ]; then if [ -f "$config_file" ]; then
eval "$config_file" # shellcheck disable=SC1090
source "$config_file"
config_msg "Sourced" "$config_file" config_msg "Sourced" "$config_file"
else else
msg "Config file $config_file not found" msg "Config file $config_file not found"

View File

@@ -100,8 +100,6 @@ fetch_and_filter_data()
echo "$data" echo "$data"
} }
# Format file-specific comments grouped by review # Format file-specific comments grouped by review
format_grouped_review_comments() format_grouped_review_comments()
{ {

2
local/bin/x-ssh-audit Normal file → Executable file
View File

@@ -350,7 +350,7 @@ ssh_with_retry()
log_message "INFO" "Trying SSH agent/default authentication" log_message "INFO" "Trying SSH agent/default authentication"
fi fi
# shellcheck disable=SC2086 # shellcheck disable=SC2086,SC2029
if ssh $ssh_opts "${username}@${host}" "$command" 2> /dev/null; then if ssh $ssh_opts "${username}@${host}" "$command" 2> /dev/null; then
if [ "$try_key" != "NO_KEY" ]; then if [ "$try_key" != "NO_KEY" ]; then
log_message "SUCCESS" "Connected using SSH key: $try_key" log_message "SUCCESS" "Connected using SSH key: $try_key"

View File

@@ -52,17 +52,18 @@ fi
for name in "$@"; do for name in "$@"; do
# Make a temporary file # Make a temporary file
# Test if we have BSD or GNU version of mktemp # Try GNU syntax first, fall back to BSD
if (strings "$(which mktemp)" | grep -q GNU); then if tmp=$(mktemp 2> /dev/null) && [ -f "$tmp" ]; then
# We have the GNU version # GNU mktemp succeeded
tmp=$(mktemp) :
else else
# We have the BSD version # Try BSD syntax
tmp=$(mktemp -t tmp) tmp=$(mktemp -t tmp)
fi fi
# Download the certificate # Download the certificate
if (! echo "" | openssl s_client -connect "$name:$port" > "$tmp" 2> /dev/null); then if (! echo "" | openssl s_client -connect "$name:$port" > "$tmp" 2> /dev/null); then
rm -f "$tmp"
echo "Failed to get cert from https://$name:$port/" echo "Failed to get cert from https://$name:$port/"
exit 3 exit 3
fi fi
@@ -74,26 +75,26 @@ for name in "$@"; do
rm -f "$tmp" rm -f "$tmp"
# Convert the expiry date + todays date to seconds-past epoch # Convert the expiry date + todays date to seconds-past epoch
# Check if we have the BSD or the GNU version of date # Try GNU syntax first, fall back to BSD
if (strings "$(which date)" | grep -q GNU); then if then=$(date --date "$date" +%s 2> /dev/null); then
# We have GNU this is easy # GNU date succeeded
then=$(date --date "$date" +%s) :
else else
# We have BSD now it is getting complicated # BSD date requires manual parsing
year=$(echo "$date" | awk '{print $4}') year=$(echo "$date" | awk '{print $4}')
month=$(echo "$date" | awk '{print $1}') month=$(echo "$date" | awk '{print $1}')
day=$(echo "$date" | awk '{print $2}') day=$(echo "$date" | awk '{print $2}')
hour=$(echo "$date" | awk '{print $3}' | awk -F: '{print $1}') hour=$(echo "$date" | awk '{print $3}' | awk -F: '{print $1}')
minute=$(echo "$date" | awk '{print $3}' | awk -F: '{print $2}') minute=$(echo "$date" | awk '{print $3}' | awk -F: '{print $2}')
second=$(echo "$date" | awk '{print $3}' | awk -F: '{print $3}') second=$(echo "$date" | awk '{print $3}' | awk -F: '{print $3}')
then=$(date -v${year}y -v${month} -v${day}d -v${hour}H -v${minute}M -v${second}S -u +%s) then=$(date -v"${year}"y -v"${month}" -v"${day}"d -v"${hour}"H -v"${minute}"M -v"${second}"S -u +%s)
fi fi
now=$(date +%s) now=$(date +%s)
# Day diff # Day diff
diff=$(("$then" - "$now")) diff=$(("$then" - "$now"))
diff=$($diff / 86400) diff=$((diff / 86400))
# All done # All done
if [ "$days" = "1" ]; then if [ "$days" = "1" ]; then

View File

@@ -24,10 +24,10 @@ resetOutput()
# ranging between 0 and 255 inclusive # ranging between 0 and 255 inclusive
rainbowColor() rainbowColor()
{ {
let h=$1/43 ((h = $1 / 43))
let f=$1-43*$h ((f = $1 - 43 * h))
let t=$f*255/43 ((t = f * 255 / 43))
let q=255-t ((q = 255 - t))
if [ $h -eq 0 ]; then if [ $h -eq 0 ]; then
echo "255 $t 0" echo "255 $t 0"
@@ -81,11 +81,13 @@ done
resetOutput resetOutput
for i in $(seq 0 127); do for i in $(seq 0 127); do
# shellcheck disable=SC2046 # Intentional word splitting to pass 3 RGB values as separate args
setBackgroundColor $(rainbowColor "$i") setBackgroundColor $(rainbowColor "$i")
echo -n " " echo -n " "
done done
resetOutput resetOutput
for i in $(seq 255 128); do for i in $(seq 255 128); do
# shellcheck disable=SC2046 # Intentional word splitting to pass 3 RGB values as separate args
setBackgroundColor $(rainbowColor "$i") setBackgroundColor $(rainbowColor "$i")
echo -n " " echo -n " "
done done

View File

@@ -126,8 +126,8 @@ parse_options()
# Generate thumbnails recursively using find and filtering by MIME type # Generate thumbnails recursively using find and filtering by MIME type
generate_thumbnails() generate_thumbnails()
{ {
local source_dir=$1 local source_dir="$1"
local output_dir=$2 local output_dir="$2"
# Ensure the output directory exists (create if necessary) # Ensure the output directory exists (create if necessary)
if [ ! -d "$output_dir" ]; then if [ ! -d "$output_dir" ]; then
@@ -144,7 +144,7 @@ generate_thumbnails()
fi fi
# Determine the relative path with respect to the source directory. # Determine the relative path with respect to the source directory.
rel_path="${file#$source_dir/}" rel_path="${file#"$source_dir"/}"
dir="$(dirname "$rel_path")" dir="$(dirname "$rel_path")"
base="$(basename "$rel_path")" base="$(basename "$rel_path")"
filename="${base%.*}" filename="${base%.*}"

View File

@@ -6,14 +6,13 @@
"doc": "docs" "doc": "docs"
}, },
"scripts": { "scripts": {
"lint:markdown": "markdownlint -d .", "lint:biome": "biome check .",
"fix:markdown": "markdownlint -df .", "fix:biome": "biome check --write .",
"lint:prettier": "prettier . --check", "format": "biome format --write .",
"fix:prettier": "prettier . --write",
"test": "bash test-all.sh", "test": "bash test-all.sh",
"lint:ec": "ec -f gcc", "lint:ec": "ec -f gcc",
"lint": "yarn lint:markdown && yarn lint:prettier && yarn lint:ec", "lint": "yarn lint:biome && yarn lint:ec",
"fix": "yarn fix:markdown && yarn fix:prettier" "fix": "yarn fix:biome"
}, },
"repository": { "repository": {
"type": "git", "type": "git",
@@ -30,13 +29,11 @@
}, },
"homepage": "https://github.com/ivuorinen/dotfiles#readme", "homepage": "https://github.com/ivuorinen/dotfiles#readme",
"devDependencies": { "devDependencies": {
"@ivuorinen/base-configs": "1.0.0", "@biomejs/biome": "^2.3.1",
"@types/node": "^24.10.1", "@types/node": "^24.0.1",
"bats": "^1.13.0", "bats": "^1.12.0",
"editorconfig-checker": "^6.1.1", "editorconfig-checker": "^6.1.0",
"markdownlint": "^0.40.0", "typescript": "^5.8.3"
"prettier": "^3.7.4",
"typescript": "^5.9.3"
}, },
"packageManager": "yarn@4.12.0" "packageManager": "yarn@4.12.0"
} }

653
yarn.lock
View File

@@ -5,178 +5,122 @@ __metadata:
version: 8 version: 8
cacheKey: 10c0 cacheKey: 10c0
"@ivuorinen/base-configs@npm:1.0.0": "@biomejs/biome@npm:^2.3.1":
version: 1.0.0 version: 2.3.1
resolution: "@ivuorinen/base-configs@npm:1.0.0" resolution: "@biomejs/biome@npm:2.3.1"
checksum: 10c0/399e36bb7352b9e2ad3d91a5632a312670349d7e0976336c6fe113137c6b8c97d92b0b3b05bbe26ea8b533d27406162a5d42889019d27657efe7f08dbb54d820
languageName: node
linkType: hard
"@types/debug@npm:^4.0.0":
version: 4.1.12
resolution: "@types/debug@npm:4.1.12"
dependencies: dependencies:
"@types/ms": "npm:*" "@biomejs/cli-darwin-arm64": "npm:2.3.1"
checksum: 10c0/5dcd465edbb5a7f226e9a5efd1f399c6172407ef5840686b73e3608ce135eeca54ae8037dcd9f16bdb2768ac74925b820a8b9ecc588a58ca09eca6acabe33e2f "@biomejs/cli-darwin-x64": "npm:2.3.1"
"@biomejs/cli-linux-arm64": "npm:2.3.1"
"@biomejs/cli-linux-arm64-musl": "npm:2.3.1"
"@biomejs/cli-linux-x64": "npm:2.3.1"
"@biomejs/cli-linux-x64-musl": "npm:2.3.1"
"@biomejs/cli-win32-arm64": "npm:2.3.1"
"@biomejs/cli-win32-x64": "npm:2.3.1"
dependenciesMeta:
"@biomejs/cli-darwin-arm64":
optional: true
"@biomejs/cli-darwin-x64":
optional: true
"@biomejs/cli-linux-arm64":
optional: true
"@biomejs/cli-linux-arm64-musl":
optional: true
"@biomejs/cli-linux-x64":
optional: true
"@biomejs/cli-linux-x64-musl":
optional: true
"@biomejs/cli-win32-arm64":
optional: true
"@biomejs/cli-win32-x64":
optional: true
bin:
biome: bin/biome
checksum: 10c0/3c644c095b4e9be54b8e93275d8a78bed766a3781b7b4c5c97fd1a736ae6bb3f1f440daa8ceaa1f8f929af711fcffd1cf754f5149315a4695e2b868e29f5d293
languageName: node languageName: node
linkType: hard linkType: hard
"@types/katex@npm:^0.16.0": "@biomejs/cli-darwin-arm64@npm:2.3.1":
version: 0.16.7 version: 2.3.1
resolution: "@types/katex@npm:0.16.7" resolution: "@biomejs/cli-darwin-arm64@npm:2.3.1"
checksum: 10c0/68dcb9f68a90513ec78ca0196a142e15c2a2c270b1520d752bafd47a99207115085a64087b50140359017d7e9c870b3c68e7e4d36668c9e348a9ef0c48919b5a conditions: os=darwin & cpu=arm64
languageName: node languageName: node
linkType: hard linkType: hard
"@types/ms@npm:*": "@biomejs/cli-darwin-x64@npm:2.3.1":
version: 2.1.0 version: 2.3.1
resolution: "@types/ms@npm:2.1.0" resolution: "@biomejs/cli-darwin-x64@npm:2.3.1"
checksum: 10c0/5ce692ffe1549e1b827d99ef8ff71187457e0eb44adbae38fdf7b9a74bae8d20642ee963c14516db1d35fa2652e65f47680fdf679dcbde52bbfadd021f497225 conditions: os=darwin & cpu=x64
languageName: node languageName: node
linkType: hard linkType: hard
"@types/node@npm:^24.10.1": "@biomejs/cli-linux-arm64-musl@npm:2.3.1":
version: 24.10.1 version: 2.3.1
resolution: "@types/node@npm:24.10.1" resolution: "@biomejs/cli-linux-arm64-musl@npm:2.3.1"
conditions: os=linux & cpu=arm64 & libc=musl
languageName: node
linkType: hard
"@biomejs/cli-linux-arm64@npm:2.3.1":
version: 2.3.1
resolution: "@biomejs/cli-linux-arm64@npm:2.3.1"
conditions: os=linux & cpu=arm64 & libc=glibc
languageName: node
linkType: hard
"@biomejs/cli-linux-x64-musl@npm:2.3.1":
version: 2.3.1
resolution: "@biomejs/cli-linux-x64-musl@npm:2.3.1"
conditions: os=linux & cpu=x64 & libc=musl
languageName: node
linkType: hard
"@biomejs/cli-linux-x64@npm:2.3.1":
version: 2.3.1
resolution: "@biomejs/cli-linux-x64@npm:2.3.1"
conditions: os=linux & cpu=x64 & libc=glibc
languageName: node
linkType: hard
"@biomejs/cli-win32-arm64@npm:2.3.1":
version: 2.3.1
resolution: "@biomejs/cli-win32-arm64@npm:2.3.1"
conditions: os=win32 & cpu=arm64
languageName: node
linkType: hard
"@biomejs/cli-win32-x64@npm:2.3.1":
version: 2.3.1
resolution: "@biomejs/cli-win32-x64@npm:2.3.1"
conditions: os=win32 & cpu=x64
languageName: node
linkType: hard
"@types/node@npm:^24.0.1":
version: 24.2.1
resolution: "@types/node@npm:24.2.1"
dependencies: dependencies:
undici-types: "npm:~7.16.0" undici-types: "npm:~7.10.0"
checksum: 10c0/d6bca7a78f550fbb376f236f92b405d676003a8a09a1b411f55920ef34286ee3ee51f566203920e835478784df52662b5b2af89159d9d319352e9ea21801c002 checksum: 10c0/439a3c7edf88a298e0c92e46f670234070b892589c3b06e82cc86c47a7e1cf220f4a4b4736ec6ac7e4b9e1c40d7b6d443a1e22f99dd17f13f9dd15de3b32011b
languageName: node languageName: node
linkType: hard linkType: hard
"@types/unist@npm:^2.0.0": "bats@npm:^1.12.0":
version: 2.0.11 version: 1.12.0
resolution: "@types/unist@npm:2.0.11" resolution: "bats@npm:1.12.0"
checksum: 10c0/24dcdf25a168f453bb70298145eb043cfdbb82472db0bc0b56d6d51cd2e484b9ed8271d4ac93000a80da568f2402e9339723db262d0869e2bf13bc58e081768d
languageName: node
linkType: hard
"ansi-regex@npm:^6.0.1":
version: 6.2.2
resolution: "ansi-regex@npm:6.2.2"
checksum: 10c0/05d4acb1d2f59ab2cf4b794339c7b168890d44dda4bf0ce01152a8da0213aca207802f930442ce8cd22d7a92f44907664aac6508904e75e038fa944d2601b30f
languageName: node
linkType: hard
"bats@npm:^1.13.0":
version: 1.13.0
resolution: "bats@npm:1.13.0"
bin: bin:
bats: bin/bats bats: bin/bats
checksum: 10c0/7f697d6305d80d328d620bd58c658f7830fbed57e28a43f18771d89fb6f941a131b440ab18d951fd15fdd23f3c687687d607d7fc03ac4c99f4725e64d2432832 checksum: 10c0/fa601c7851e0e8bd773987e5b103e48348c63c630432a1d80e0f1849ca0133b22b3eab4f148566dac80b51b42e44db6c54f86a1021b87832a648f38b2501e3a8
languageName: node languageName: node
linkType: hard linkType: hard
"character-entities-legacy@npm:^3.0.0": "editorconfig-checker@npm:^6.1.0":
version: 3.0.0 version: 6.1.0
resolution: "character-entities-legacy@npm:3.0.0" resolution: "editorconfig-checker@npm:6.1.0"
checksum: 10c0/ec4b430af873661aa754a896a2b55af089b4e938d3d010fad5219299a6b6d32ab175142699ee250640678cd64bdecd6db3c9af0b8759ab7b155d970d84c4c7d1
languageName: node
linkType: hard
"character-entities@npm:^2.0.0":
version: 2.0.2
resolution: "character-entities@npm:2.0.2"
checksum: 10c0/b0c645a45bcc90ff24f0e0140f4875a8436b8ef13b6bcd31ec02cfb2ca502b680362aa95386f7815bdc04b6464d48cf191210b3840d7c04241a149ede591a308
languageName: node
linkType: hard
"character-reference-invalid@npm:^2.0.0":
version: 2.0.1
resolution: "character-reference-invalid@npm:2.0.1"
checksum: 10c0/2ae0dec770cd8659d7e8b0ce24392d83b4c2f0eb4a3395c955dce5528edd4cc030a794cfa06600fcdd700b3f2de2f9b8e40e309c0011c4180e3be64a0b42e6a1
languageName: node
linkType: hard
"commander@npm:^8.3.0":
version: 8.3.0
resolution: "commander@npm:8.3.0"
checksum: 10c0/8b043bb8322ea1c39664a1598a95e0495bfe4ca2fad0d84a92d7d1d8d213e2a155b441d2470c8e08de7c4a28cf2bc6e169211c49e1b21d9f7edc6ae4d9356060
languageName: node
linkType: hard
"debug@npm:^4.0.0":
version: 4.4.1
resolution: "debug@npm:4.4.1"
dependencies:
ms: "npm:^2.1.3"
peerDependenciesMeta:
supports-color:
optional: true
checksum: 10c0/d2b44bc1afd912b49bb7ebb0d50a860dc93a4dd7d946e8de94abc957bb63726b7dd5aa48c18c2386c379ec024c46692e15ed3ed97d481729f929201e671fcd55
languageName: node
linkType: hard
"decode-named-character-reference@npm:^1.0.0":
version: 1.2.0
resolution: "decode-named-character-reference@npm:1.2.0"
dependencies:
character-entities: "npm:^2.0.0"
checksum: 10c0/761a89de6b0e0a2d4b21ae99074e4cc3344dd11eb29f112e23cc5909f2e9f33c5ed20cd6b146b27fb78170bce0f3f9b3362a84b75638676a05c938c24a60f5d7
languageName: node
linkType: hard
"dequal@npm:^2.0.0":
version: 2.0.3
resolution: "dequal@npm:2.0.3"
checksum: 10c0/f98860cdf58b64991ae10205137c0e97d384c3a4edc7f807603887b7c4b850af1224a33d88012009f150861cbee4fa2d322c4cc04b9313bee312e47f6ecaa888
languageName: node
linkType: hard
"devlop@npm:^1.0.0":
version: 1.1.0
resolution: "devlop@npm:1.1.0"
dependencies:
dequal: "npm:^2.0.0"
checksum: 10c0/e0928ab8f94c59417a2b8389c45c55ce0a02d9ac7fd74ef62d01ba48060129e1d594501b77de01f3eeafc7cb00773819b0df74d96251cf20b31c5b3071f45c0e
languageName: node
linkType: hard
"editorconfig-checker@npm:^6.1.1":
version: 6.1.1
resolution: "editorconfig-checker@npm:6.1.1"
bin: bin:
ec: dist/index.js ec: dist/index.js
editorconfig-checker: dist/index.js editorconfig-checker: dist/index.js
checksum: 10c0/0a46ce93e2821041c4b4bbf2ab9dc30e1b4eb03d3f20e5b14bbe45767f6f2aafd5e1310ea737c15402d8e193f702a421a814041e072584250e8a7d5e63d83741 checksum: 10c0/0876f5adeaf8c40dd179d78449bd3f14cf428f9179714cc49614407a525c185c228173ba849ff81c340af39a7c456cb904598bf3b60407b28b8302f2672ddb91
languageName: node
linkType: hard
"get-east-asian-width@npm:^1.3.0":
version: 1.4.0
resolution: "get-east-asian-width@npm:1.4.0"
checksum: 10c0/4e481d418e5a32061c36fbb90d1b225a254cc5b2df5f0b25da215dcd335a3c111f0c2023ffda43140727a9cafb62dac41d022da82c08f31083ee89f714ee3b83
languageName: node
linkType: hard
"is-alphabetical@npm:^2.0.0":
version: 2.0.1
resolution: "is-alphabetical@npm:2.0.1"
checksum: 10c0/932367456f17237533fd1fc9fe179df77957271020b83ea31da50e5cc472d35ef6b5fb8147453274ffd251134472ce24eb6f8d8398d96dee98237cdb81a6c9a7
languageName: node
linkType: hard
"is-alphanumerical@npm:^2.0.0":
version: 2.0.1
resolution: "is-alphanumerical@npm:2.0.1"
dependencies:
is-alphabetical: "npm:^2.0.0"
is-decimal: "npm:^2.0.0"
checksum: 10c0/4b35c42b18e40d41378293f82a3ecd9de77049b476f748db5697c297f686e1e05b072a6aaae2d16f54d2a57f85b00cbbe755c75f6d583d1c77d6657bd0feb5a2
languageName: node
linkType: hard
"is-decimal@npm:^2.0.0":
version: 2.0.1
resolution: "is-decimal@npm:2.0.1"
checksum: 10c0/8085dd66f7d82f9de818fba48b9e9c0429cb4291824e6c5f2622e96b9680b54a07a624cfc663b24148b8e853c62a1c987cfe8b0b5a13f5156991afaf6736e334
languageName: node
linkType: hard
"is-hexadecimal@npm:^2.0.0":
version: 2.0.1
resolution: "is-hexadecimal@npm:2.0.1"
checksum: 10c0/3eb60fe2f1e2bbc760b927dcad4d51eaa0c60138cf7fc671803f66353ad90c301605b502c7ea4c6bb0548e1c7e79dfd37b73b632652e3b76030bba603a7e9626
languageName: node languageName: node
linkType: hard linkType: hard
@@ -184,412 +128,37 @@ __metadata:
version: 0.0.0-use.local version: 0.0.0-use.local
resolution: "ivuorinen-dotfiles@workspace:." resolution: "ivuorinen-dotfiles@workspace:."
dependencies: dependencies:
"@ivuorinen/base-configs": "npm:1.0.0" "@biomejs/biome": "npm:^2.3.1"
"@types/node": "npm:^24.10.1" "@types/node": "npm:^24.0.1"
bats: "npm:^1.13.0" bats: "npm:^1.12.0"
editorconfig-checker: "npm:^6.1.1" editorconfig-checker: "npm:^6.1.0"
markdownlint: "npm:^0.40.0" typescript: "npm:^5.8.3"
prettier: "npm:^3.7.4"
typescript: "npm:^5.9.3"
languageName: unknown languageName: unknown
linkType: soft linkType: soft
"katex@npm:^0.16.0": "typescript@npm:^5.8.3":
version: 0.16.22 version: 5.9.2
resolution: "katex@npm:0.16.22" resolution: "typescript@npm:5.9.2"
dependencies:
commander: "npm:^8.3.0"
bin:
katex: cli.js
checksum: 10c0/07b8b1f07ae53171b5f1ea0cf6f18841d2055825c8b11cd81cfe039afcd3af2cfc84ad033531ee3875088329105195b039c267e0dd4b0c237807e3c3b2009913
languageName: node
linkType: hard
"markdownlint@npm:^0.40.0":
version: 0.40.0
resolution: "markdownlint@npm:0.40.0"
dependencies:
micromark: "npm:4.0.2"
micromark-core-commonmark: "npm:2.0.3"
micromark-extension-directive: "npm:4.0.0"
micromark-extension-gfm-autolink-literal: "npm:2.1.0"
micromark-extension-gfm-footnote: "npm:2.1.0"
micromark-extension-gfm-table: "npm:2.1.1"
micromark-extension-math: "npm:3.1.0"
micromark-util-types: "npm:2.0.2"
string-width: "npm:8.1.0"
checksum: 10c0/1543fcf4a433bc54e0e565cb1c8111e5e3d0df3742df0cc840d470bced21a1e3b5593e4e380ad0d8d5e490d9b399699d48aeabed33719f3fbdc6d00128138f20
languageName: node
linkType: hard
"micromark-core-commonmark@npm:2.0.3, micromark-core-commonmark@npm:^2.0.0":
version: 2.0.3
resolution: "micromark-core-commonmark@npm:2.0.3"
dependencies:
decode-named-character-reference: "npm:^1.0.0"
devlop: "npm:^1.0.0"
micromark-factory-destination: "npm:^2.0.0"
micromark-factory-label: "npm:^2.0.0"
micromark-factory-space: "npm:^2.0.0"
micromark-factory-title: "npm:^2.0.0"
micromark-factory-whitespace: "npm:^2.0.0"
micromark-util-character: "npm:^2.0.0"
micromark-util-chunked: "npm:^2.0.0"
micromark-util-classify-character: "npm:^2.0.0"
micromark-util-html-tag-name: "npm:^2.0.0"
micromark-util-normalize-identifier: "npm:^2.0.0"
micromark-util-resolve-all: "npm:^2.0.0"
micromark-util-subtokenize: "npm:^2.0.0"
micromark-util-symbol: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/bd4a794fdc9e88dbdf59eaf1c507ddf26e5f7ddf4e52566c72239c0f1b66adbcd219ba2cd42350debbe24471434d5f5e50099d2b3f4e5762ca222ba8e5b549ee
languageName: node
linkType: hard
"micromark-extension-directive@npm:4.0.0":
version: 4.0.0
resolution: "micromark-extension-directive@npm:4.0.0"
dependencies:
devlop: "npm:^1.0.0"
micromark-factory-space: "npm:^2.0.0"
micromark-factory-whitespace: "npm:^2.0.0"
micromark-util-character: "npm:^2.0.0"
micromark-util-symbol: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
parse-entities: "npm:^4.0.0"
checksum: 10c0/b4aef0f44339543466ae186130a4514985837b6b12d0c155bd1162e740f631e58f0883a39d0c723206fa0ff53a9b579965c79116f902236f6f123c3340b5fefb
languageName: node
linkType: hard
"micromark-extension-gfm-autolink-literal@npm:2.1.0":
version: 2.1.0
resolution: "micromark-extension-gfm-autolink-literal@npm:2.1.0"
dependencies:
micromark-util-character: "npm:^2.0.0"
micromark-util-sanitize-uri: "npm:^2.0.0"
micromark-util-symbol: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/84e6fbb84ea7c161dfa179665dc90d51116de4c28f3e958260c0423e5a745372b7dcbc87d3cde98213b532e6812f847eef5ae561c9397d7f7da1e59872ef3efe
languageName: node
linkType: hard
"micromark-extension-gfm-footnote@npm:2.1.0":
version: 2.1.0
resolution: "micromark-extension-gfm-footnote@npm:2.1.0"
dependencies:
devlop: "npm:^1.0.0"
micromark-core-commonmark: "npm:^2.0.0"
micromark-factory-space: "npm:^2.0.0"
micromark-util-character: "npm:^2.0.0"
micromark-util-normalize-identifier: "npm:^2.0.0"
micromark-util-sanitize-uri: "npm:^2.0.0"
micromark-util-symbol: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/d172e4218968b7371b9321af5cde8c77423f73b233b2b0fcf3ff6fd6f61d2e0d52c49123a9b7910612478bf1f0d5e88c75a3990dd68f70f3933fe812b9f77edc
languageName: node
linkType: hard
"micromark-extension-gfm-table@npm:2.1.1":
version: 2.1.1
resolution: "micromark-extension-gfm-table@npm:2.1.1"
dependencies:
devlop: "npm:^1.0.0"
micromark-factory-space: "npm:^2.0.0"
micromark-util-character: "npm:^2.0.0"
micromark-util-symbol: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/04bc00e19b435fa0add62cd029d8b7eb6137522f77832186b1d5ef34544a9bd030c9cf85e92ddfcc5c31f6f0a58a43d4b96dba4fc21316037c734630ee12c912
languageName: node
linkType: hard
"micromark-extension-math@npm:3.1.0":
version: 3.1.0
resolution: "micromark-extension-math@npm:3.1.0"
dependencies:
"@types/katex": "npm:^0.16.0"
devlop: "npm:^1.0.0"
katex: "npm:^0.16.0"
micromark-factory-space: "npm:^2.0.0"
micromark-util-character: "npm:^2.0.0"
micromark-util-symbol: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/56e6f2185a4613f9d47e7e98cf8605851c990957d9229c942b005e286c8087b61dc9149448d38b2f8be6d42cc6a64aad7e1f2778ddd86fbbb1a2f48a3ca1872f
languageName: node
linkType: hard
"micromark-factory-destination@npm:^2.0.0":
version: 2.0.1
resolution: "micromark-factory-destination@npm:2.0.1"
dependencies:
micromark-util-character: "npm:^2.0.0"
micromark-util-symbol: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/bbafcf869cee5bf511161354cb87d61c142592fbecea051000ff116068dc85216e6d48519d147890b9ea5d7e2864a6341c0c09d9948c203bff624a80a476023c
languageName: node
linkType: hard
"micromark-factory-label@npm:^2.0.0":
version: 2.0.1
resolution: "micromark-factory-label@npm:2.0.1"
dependencies:
devlop: "npm:^1.0.0"
micromark-util-character: "npm:^2.0.0"
micromark-util-symbol: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/0137716b4ecb428114165505e94a2f18855c8bbea21b07a8b5ce514b32a595ed789d2b967125718fc44c4197ceaa48f6609d58807a68e778138d2e6b91b824e8
languageName: node
linkType: hard
"micromark-factory-space@npm:^2.0.0":
version: 2.0.1
resolution: "micromark-factory-space@npm:2.0.1"
dependencies:
micromark-util-character: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/f9ed43f1c0652d8d898de0ac2be3f77f776fffe7dd96bdbba1e02d7ce33d3853c6ff5daa52568fc4fa32cdf3a62d86b85ead9b9189f7211e1d69ff2163c450fb
languageName: node
linkType: hard
"micromark-factory-title@npm:^2.0.0":
version: 2.0.1
resolution: "micromark-factory-title@npm:2.0.1"
dependencies:
micromark-factory-space: "npm:^2.0.0"
micromark-util-character: "npm:^2.0.0"
micromark-util-symbol: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/e72fad8d6e88823514916890099a5af20b6a9178ccf78e7e5e05f4de99bb8797acb756257d7a3a57a53854cb0086bf8aab15b1a9e9db8982500dd2c9ff5948b6
languageName: node
linkType: hard
"micromark-factory-whitespace@npm:^2.0.0":
version: 2.0.1
resolution: "micromark-factory-whitespace@npm:2.0.1"
dependencies:
micromark-factory-space: "npm:^2.0.0"
micromark-util-character: "npm:^2.0.0"
micromark-util-symbol: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/20a1ec58698f24b766510a309b23a10175034fcf1551eaa9da3adcbed3e00cd53d1ebe5f030cf873f76a1cec3c34eb8c50cc227be3344caa9ed25d56cf611224
languageName: node
linkType: hard
"micromark-util-character@npm:^2.0.0":
version: 2.1.1
resolution: "micromark-util-character@npm:2.1.1"
dependencies:
micromark-util-symbol: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/d3fe7a5e2c4060fc2a076f9ce699c82a2e87190a3946e1e5eea77f563869b504961f5668d9c9c014724db28ac32fa909070ea8b30c3a39bd0483cc6c04cc76a1
languageName: node
linkType: hard
"micromark-util-chunked@npm:^2.0.0":
version: 2.0.1
resolution: "micromark-util-chunked@npm:2.0.1"
dependencies:
micromark-util-symbol: "npm:^2.0.0"
checksum: 10c0/b68c0c16fe8106949537bdcfe1be9cf36c0ccd3bc54c4007003cb0984c3750b6cdd0fd77d03f269a3382b85b0de58bde4f6eedbe7ecdf7244759112289b1ab56
languageName: node
linkType: hard
"micromark-util-classify-character@npm:^2.0.0":
version: 2.0.1
resolution: "micromark-util-classify-character@npm:2.0.1"
dependencies:
micromark-util-character: "npm:^2.0.0"
micromark-util-symbol: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/8a02e59304005c475c332f581697e92e8c585bcd45d5d225a66c1c1b14ab5a8062705188c2ccec33cc998d33502514121478b2091feddbc751887fc9c290ed08
languageName: node
linkType: hard
"micromark-util-combine-extensions@npm:^2.0.0":
version: 2.0.1
resolution: "micromark-util-combine-extensions@npm:2.0.1"
dependencies:
micromark-util-chunked: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/f15e282af24c8372cbb10b9b0b3e2c0aa681fea0ca323a44d6bc537dc1d9382c819c3689f14eaa000118f5a163245358ce6276b2cda9a84439cdb221f5d86ae7
languageName: node
linkType: hard
"micromark-util-decode-numeric-character-reference@npm:^2.0.0":
version: 2.0.2
resolution: "micromark-util-decode-numeric-character-reference@npm:2.0.2"
dependencies:
micromark-util-symbol: "npm:^2.0.0"
checksum: 10c0/9c8a9f2c790e5593ffe513901c3a110e9ec8882a08f466da014112a25e5059b51551ca0aeb7ff494657d86eceb2f02ee556c6558b8d66aadc61eae4a240da0df
languageName: node
linkType: hard
"micromark-util-encode@npm:^2.0.0":
version: 2.0.1
resolution: "micromark-util-encode@npm:2.0.1"
checksum: 10c0/b2b29f901093845da8a1bf997ea8b7f5e061ffdba85070dfe14b0197c48fda64ffcf82bfe53c90cf9dc185e69eef8c5d41cae3ba918b96bc279326921b59008a
languageName: node
linkType: hard
"micromark-util-html-tag-name@npm:^2.0.0":
version: 2.0.1
resolution: "micromark-util-html-tag-name@npm:2.0.1"
checksum: 10c0/ae80444db786fde908e9295f19a27a4aa304171852c77414516418650097b8afb401961c9edb09d677b06e97e8370cfa65638dde8438ebd41d60c0a8678b85b9
languageName: node
linkType: hard
"micromark-util-normalize-identifier@npm:^2.0.0":
version: 2.0.1
resolution: "micromark-util-normalize-identifier@npm:2.0.1"
dependencies:
micromark-util-symbol: "npm:^2.0.0"
checksum: 10c0/5299265fa360769fc499a89f40142f10a9d4a5c3dd8e6eac8a8ef3c2e4a6570e4c009cf75ea46dce5ee31c01f25587bde2f4a5cc0a935584ae86dd857f2babbd
languageName: node
linkType: hard
"micromark-util-resolve-all@npm:^2.0.0":
version: 2.0.1
resolution: "micromark-util-resolve-all@npm:2.0.1"
dependencies:
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/bb6ca28764696bb479dc44a2d5b5fe003e7177aeae1d6b0d43f24cc223bab90234092d9c3ce4a4d2b8df095ccfd820537b10eb96bb7044d635f385d65a4c984a
languageName: node
linkType: hard
"micromark-util-sanitize-uri@npm:^2.0.0":
version: 2.0.1
resolution: "micromark-util-sanitize-uri@npm:2.0.1"
dependencies:
micromark-util-character: "npm:^2.0.0"
micromark-util-encode: "npm:^2.0.0"
micromark-util-symbol: "npm:^2.0.0"
checksum: 10c0/60e92166e1870fd4f1961468c2651013ff760617342918e0e0c3c4e872433aa2e60c1e5a672bfe5d89dc98f742d6b33897585cf86ae002cda23e905a3c02527c
languageName: node
linkType: hard
"micromark-util-subtokenize@npm:^2.0.0":
version: 2.1.0
resolution: "micromark-util-subtokenize@npm:2.1.0"
dependencies:
devlop: "npm:^1.0.0"
micromark-util-chunked: "npm:^2.0.0"
micromark-util-symbol: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/bee69eece4393308e657c293ba80d92ebcb637e5f55e21dcf9c3fa732b91a8eda8ac248d76ff375e675175bfadeae4712e5158ef97eef1111789da1ce7ab5067
languageName: node
linkType: hard
"micromark-util-symbol@npm:^2.0.0":
version: 2.0.1
resolution: "micromark-util-symbol@npm:2.0.1"
checksum: 10c0/f2d1b207771e573232436618e78c5e46cd4b5c560dd4a6d63863d58018abbf49cb96ec69f7007471e51434c60de3c9268ef2bf46852f26ff4aacd10f9da16fe9
languageName: node
linkType: hard
"micromark-util-types@npm:2.0.2, micromark-util-types@npm:^2.0.0":
version: 2.0.2
resolution: "micromark-util-types@npm:2.0.2"
checksum: 10c0/c8c15b96c858db781c4393f55feec10004bf7df95487636c9a9f7209e51002a5cca6a047c5d2a5dc669ff92da20e57aaa881e81a268d9ccadb647f9dce305298
languageName: node
linkType: hard
"micromark@npm:4.0.2":
version: 4.0.2
resolution: "micromark@npm:4.0.2"
dependencies:
"@types/debug": "npm:^4.0.0"
debug: "npm:^4.0.0"
decode-named-character-reference: "npm:^1.0.0"
devlop: "npm:^1.0.0"
micromark-core-commonmark: "npm:^2.0.0"
micromark-factory-space: "npm:^2.0.0"
micromark-util-character: "npm:^2.0.0"
micromark-util-chunked: "npm:^2.0.0"
micromark-util-combine-extensions: "npm:^2.0.0"
micromark-util-decode-numeric-character-reference: "npm:^2.0.0"
micromark-util-encode: "npm:^2.0.0"
micromark-util-normalize-identifier: "npm:^2.0.0"
micromark-util-resolve-all: "npm:^2.0.0"
micromark-util-sanitize-uri: "npm:^2.0.0"
micromark-util-subtokenize: "npm:^2.0.0"
micromark-util-symbol: "npm:^2.0.0"
micromark-util-types: "npm:^2.0.0"
checksum: 10c0/07462287254219d6eda6eac8a3cebaff2994e0575499e7088027b825105e096e4f51e466b14b2a81b71933a3b6c48ee069049d87bc2c2127eee50d9cc69e8af6
languageName: node
linkType: hard
"ms@npm:^2.1.3":
version: 2.1.3
resolution: "ms@npm:2.1.3"
checksum: 10c0/d924b57e7312b3b63ad21fc5b3dc0af5e78d61a1fc7cfb5457edaf26326bf62be5307cc87ffb6862ef1c2b33b0233cdb5d4f01c4c958cc0d660948b65a287a48
languageName: node
linkType: hard
"parse-entities@npm:^4.0.0":
version: 4.0.2
resolution: "parse-entities@npm:4.0.2"
dependencies:
"@types/unist": "npm:^2.0.0"
character-entities-legacy: "npm:^3.0.0"
character-reference-invalid: "npm:^2.0.0"
decode-named-character-reference: "npm:^1.0.0"
is-alphanumerical: "npm:^2.0.0"
is-decimal: "npm:^2.0.0"
is-hexadecimal: "npm:^2.0.0"
checksum: 10c0/a13906b1151750b78ed83d386294066daf5fb559e08c5af9591b2d98cc209123103016a01df776f65f8219ad26652d6d6b210d0974d452049cddfc53a8916c34
languageName: node
linkType: hard
"prettier@npm:^3.7.4":
version: 3.7.4
resolution: "prettier@npm:3.7.4"
bin:
prettier: bin/prettier.cjs
checksum: 10c0/9675d2cd08eacb1faf1d1a2dbfe24bfab6a912b059fc9defdb380a408893d88213e794a40a2700bd29b140eb3172e0b07c852853f6e22f16f3374659a1a13389
languageName: node
linkType: hard
"string-width@npm:8.1.0":
version: 8.1.0
resolution: "string-width@npm:8.1.0"
dependencies:
get-east-asian-width: "npm:^1.3.0"
strip-ansi: "npm:^7.1.0"
checksum: 10c0/749b5d0dab2532b4b6b801064230f4da850f57b3891287023117ab63a464ad79dd208f42f793458f48f3ad121fe2e1f01dd525ff27ead957ed9f205e27406593
languageName: node
linkType: hard
"strip-ansi@npm:^7.1.0":
version: 7.1.2
resolution: "strip-ansi@npm:7.1.2"
dependencies:
ansi-regex: "npm:^6.0.1"
checksum: 10c0/0d6d7a023de33368fd042aab0bf48f4f4077abdfd60e5393e73c7c411e85e1b3a83507c11af2e656188511475776215df9ca589b4da2295c9455cc399ce1858b
languageName: node
linkType: hard
"typescript@npm:^5.9.3":
version: 5.9.3
resolution: "typescript@npm:5.9.3"
bin: bin:
tsc: bin/tsc tsc: bin/tsc
tsserver: bin/tsserver tsserver: bin/tsserver
checksum: 10c0/6bd7552ce39f97e711db5aa048f6f9995b53f1c52f7d8667c1abdc1700c68a76a308f579cd309ce6b53646deb4e9a1be7c813a93baaf0a28ccd536a30270e1c5 checksum: 10c0/cd635d50f02d6cf98ed42de2f76289701c1ec587a363369255f01ed15aaf22be0813226bff3c53e99d971f9b540e0b3cc7583dbe05faded49b1b0bed2f638a18
languageName: node languageName: node
linkType: hard linkType: hard
"typescript@patch:typescript@npm%3A^5.9.3#optional!builtin<compat/typescript>": "typescript@patch:typescript@npm%3A^5.8.3#optional!builtin<compat/typescript>":
version: 5.9.3 version: 5.9.2
resolution: "typescript@patch:typescript@npm%3A5.9.3#optional!builtin<compat/typescript>::version=5.9.3&hash=5786d5" resolution: "typescript@patch:typescript@npm%3A5.9.2#optional!builtin<compat/typescript>::version=5.9.2&hash=5786d5"
bin: bin:
tsc: bin/tsc tsc: bin/tsc
tsserver: bin/tsserver tsserver: bin/tsserver
checksum: 10c0/ad09fdf7a756814dce65bc60c1657b40d44451346858eea230e10f2e95a289d9183b6e32e5c11e95acc0ccc214b4f36289dcad4bf1886b0adb84d711d336a430 checksum: 10c0/34d2a8e23eb8e0d1875072064d5e1d9c102e0bdce56a10a25c0b917b8aa9001a9cf5c225df12497e99da107dc379360bc138163c66b55b95f5b105b50578067e
languageName: node languageName: node
linkType: hard linkType: hard
"undici-types@npm:~7.16.0": "undici-types@npm:~7.10.0":
version: 7.16.0 version: 7.10.0
resolution: "undici-types@npm:7.16.0" resolution: "undici-types@npm:7.10.0"
checksum: 10c0/3033e2f2b5c9f1504bdc5934646cb54e37ecaca0f9249c983f7b1fc2e87c6d18399ebb05dc7fd5419e02b2e915f734d872a65da2e3eeed1813951c427d33cc9a checksum: 10c0/8b00ce50e235fe3cc601307f148b5e8fb427092ee3b23e8118ec0a5d7f68eca8cee468c8fc9f15cbb2cf2a3797945ebceb1cbd9732306a1d00e0a9b6afa0f635
languageName: node languageName: node
linkType: hard linkType: hard