compile_typst.sh
· 20 KiB · Bash
Orginalformat
#!/usr/bin/env bash
# Default values
jobs=0 # 0 means auto-detect
output_dir="pdfs"
verbose=0
quiet=0
create_dirs=0
dry_run=0
skip_newer=0
show_progress=1
force=0
select_mode=0
compile_log="typst_compile_errors.log"
move_log="typst_move_errors.log"
declare -a exclude_patterns
# Show usage/help information
show_help() {
cat << EOF
Usage: $(basename "$0") [OPTIONS]
Compile Typst files and move generated PDFs to a designated directory.
Options:
-j, --jobs NUM Number of parallel jobs (default: auto-detect)
-o, --output-dir DIR Output directory name (default: pdfs)
-v, --verbose Increase verbosity
-q, --quiet Suppress most output
-c, --create-dirs Create output directories if they don't exist
-d, --dry-run Show what would be done without doing it
-s, --skip-newer Skip compilation if PDF is newer than source
-S, --select Interactive selection mode using skim
-f, --force Force compilation even if PDF exists
--no-progress Disable progress bar
--compile-log FILE Custom location for compilation log (default: $compile_log)
--move-log FILE Custom location for move log (default: $move_log)
-e, --exclude PATTERN Exclude files matching pattern (can be used multiple times)
-h, --help Show this help message and exit
Examples:
$(basename "$0") -j 4 -o output -c
$(basename "$0") --verbose --skip-newer
$(basename "$0") -S # Select files to compile interactively
$(basename "$0") -e "**/test/**" -e "**/draft/**"
EOF
}
# Parse command line arguments
while [[ $# -gt 0 ]]; do
case $1 in
-j|--jobs)
jobs="$2"
shift 2
;;
-o|--output-dir)
output_dir="$2"
shift 2
;;
-v|--verbose)
verbose=1
shift
;;
-q|--quiet)
quiet=1
shift
;;
-c|--create-dirs)
create_dirs=1
shift
;;
-d|--dry-run)
dry_run=1
shift
;;
-s|--skip-newer)
skip_newer=1
shift
;;
-S|--select)
select_mode=1
shift
;;
-f|--force)
force=1
shift
;;
--no-progress)
show_progress=0
shift
;;
--compile-log)
compile_log="$2"
shift 2
;;
--move-log)
move_log="$2"
shift 2
;;
-e|--exclude)
exclude_patterns+=("$2")
shift 2
;;
-h|--help)
show_help
exit 0
;;
*)
echo "Unknown option: $1"
show_help
exit 1
;;
esac
done
# Check for conflicting options
if [ "$verbose" -eq 1 ] && [ "$quiet" -eq 1 ]; then
echo "Error: Cannot use both --verbose and --quiet"
exit 1
fi
# Check if required tools are installed
check_tool() {
if ! command -v "$1" &> /dev/null; then
echo "$1 is not installed. Please install it first."
echo "On most systems: $2"
exit 1
fi
}
check_tool "fd" "cargo install fd-find or apt/brew install fd-find"
check_tool "rg" "cargo install ripgrep or apt/brew install ripgrep"
check_tool "parallel" "apt/brew install parallel"
# Check for skim and bat if select mode is enabled
if [ "$select_mode" -eq 1 ]; then
check_tool "sk" "cargo install skim or apt/brew install skim"
check_tool "bat" "cargo install bat or apt/brew install bat"
fi
# ANSI color codes
GREEN='\033[0;32m'
BLUE='\033[0;34m'
YELLOW='\033[0;33m'
RED='\033[0;31m'
CYAN='\033[0;36m'
PURPLE='\033[0;35m'
NC='\033[0m' # No Color
BOLD='\033[1m'
# Nerd font icons
ICON_SUCCESS=" "
ICON_ERROR=" "
ICON_WORKING=" "
ICON_COMPILE=" "
ICON_MOVE=" "
ICON_COMPLETE=" "
ICON_SUMMARY=" "
ICON_INFO=" "
ICON_DEBUG=" "
ICON_SELECT=" "
# Logging functions
log_debug() {
if [ "$verbose" -eq 1 ]; then
echo -e "${BLUE}$ICON_DEBUG${NC} $*"
fi
}
log_info() {
if [ "$quiet" -eq 0 ]; then
echo -e "${CYAN}$ICON_INFO${NC} $*"
fi
}
log_warning() {
if [ "$quiet" -eq 0 ]; then
echo -e "${YELLOW}⚠️ ${NC} $*"
fi
}
log_error() {
echo -e "${RED}${BOLD}$ICON_ERROR${NC} $*"
}
log_success() {
if [ "$quiet" -eq 0 ]; then
echo -e "${GREEN}${BOLD}$ICON_SUCCESS${NC} $*"
fi
}
# Create a directory for temporary files
temp_dir=$(mktemp -d)
compile_failures_dir="$temp_dir/compile_failures"
move_failures_dir="$temp_dir/move_failures"
progress_dir="$temp_dir/progress"
mkdir -p "$compile_failures_dir" "$move_failures_dir" "$progress_dir"
# Create a lock file for progress updates and a flag for final progress
progress_lock="/tmp/typst_progress_lock"
final_progress_file="$temp_dir/final_progress"
touch "$final_progress_file"
# Initialize log files
> "$compile_log"
> "$move_log"
# Store current working directory
CWD=$(pwd)
log_info "Starting Typst compilation process..."
# Build exclude arguments for fd
fd_exclude_args=()
for pattern in "${exclude_patterns[@]}"; do
fd_exclude_args+=("-E" "$pattern")
done
# Create a list of files to process
typst_files_list="$temp_dir/typst_files.txt"
# Create a custom bat configuration for Typst syntax
setup_bat_for_typst() {
bat_config_dir="$temp_dir/bat_config"
mkdir -p "$bat_config_dir/syntaxes"
# Create a basic syntax mapping file for Typst
cat > "$bat_config_dir/syntaxes/typst.sublime-syntax" << 'TYPST_SYNTAX'
%YAML 1.2
---
name: Typst
file_extensions:
- typ
scope: source.typst
contexts:
main:
# Comments
- match: /\*
scope: comment.block.typst
push: block_comment
- match: //
scope: comment.line.double-slash.typst
push: line_comment
# Strings
- match: '"'
scope: punctuation.definition.string.begin.typst
push: double_string
# Math
- match: '\$'
scope: punctuation.definition.math.begin.typst
push: math
# Functions
- match: '#([a-zA-Z][a-zA-Z0-9_]*)'
scope: entity.name.function.typst
# Variables
- match: '\b([a-zA-Z][a-zA-Z0-9_]*)\s*:'
scope: variable.other.typst
# Keywords
- match: '\b(let|set|show|if|else|for|in|while|return|import|include|at|do|not|and|or|none|auto)\b'
scope: keyword.control.typst
block_comment:
- match: \*/
scope: comment.block.typst
pop: true
- match: .
scope: comment.block.typst
line_comment:
- match: $
pop: true
- match: .
scope: comment.line.double-slash.typst
double_string:
- match: '"'
scope: punctuation.definition.string.end.typst
pop: true
- match: \\.
scope: constant.character.escape.typst
- match: .
scope: string.quoted.double.typst
math:
- match: '\$'
scope: punctuation.definition.math.end.typst
pop: true
- match: .
scope: markup.math.typst
TYPST_SYNTAX
echo "$bat_config_dir"
}
if [ "$select_mode" -eq 1 ]; then
log_info "${PURPLE}$ICON_SELECT${NC} Interactive selection mode enabled"
log_info "Use TAB to select multiple files, ENTER to confirm"
# Set up bat configuration for Typst syntax highlighting
bat_config=$(setup_bat_for_typst)
export BAT_CONFIG_PATH="$bat_config"
# Use skim with bat for preview to select files
selected_files="$temp_dir/selected_files.txt"
# Find all eligible .typ files for selection
fd '\.typ$' --type f "${fd_exclude_args[@]}" . > "$typst_files_list.all"
# Check if we found any files
if [ ! -s "$typst_files_list.all" ]; then
log_error "No .typ files found matching your criteria."
rm -rf "$temp_dir"
exit 0
fi
# Prepare preview command for skim: use bat with custom syntax for .typ files
preview_cmd="bat --color=always --style=numbers --map-syntax='*.typ:Markdown' {} 2>/dev/null || cat {}"
# Use skim with bat preview to select files
cat "$typst_files_list.all" | sk --multi \
--preview "$preview_cmd" \
--preview-window "right:70%" \
--height "80%" \
--prompt "Select .typ files to compile: " \
--header "TAB: Select multiple files, ENTER: Confirm, CTRL-C: Cancel" \
--no-mouse > "$selected_files"
# Check if user selected any files
if [ ! -s "$selected_files" ]; then
log_error "No files selected. Exiting."
rm -rf "$temp_dir"
exit 0
fi
# Use the selected files instead of all discovered files
cp "$selected_files" "$typst_files_list"
total_selected=$(wc -l < "$typst_files_list")
total_available=$(wc -l < "$typst_files_list.all")
log_info "Selected ${BOLD}$total_selected${NC} out of ${BOLD}$total_available${NC} available files"
# Display tip about better Typst syntax highlighting
cat << 'TYPST_TIP'
📝 To get proper Typst syntax highlighting in bat:
1. Create a custom Typst syntax file:
mkdir -p ~/.config/bat/syntaxes
curl -L https://raw.githubusercontent.com/typst/typst-vs-code/main/syntaxes/typst.tmLanguage.json \
-o ~/.config/bat/syntaxes/typst.tmLanguage.json
2. Build bat's syntax cache:
bat cache --build
This will provide proper syntax highlighting for Typst files in future uses!
TYPST_TIP
else
# Normal mode - process all files
fd '\.typ$' --type f "${fd_exclude_args[@]}" . > "$typst_files_list"
log_info "Found ${BOLD}$(wc -l < "$typst_files_list")${NC} Typst files to process"
fi
total_files=$(wc -l < "$typst_files_list")
# Function to update progress bar during processing
update_progress_during() {
# If progress is disabled, do nothing
if [ "$show_progress" -eq 0 ]; then
return 0
fi
(
# Try to acquire lock, but don't wait if busy
flock -n 200 || return 0
completed=$(find "$progress_dir" -type f | wc -l)
percent=$((completed * 100 / total_files))
bar_length=50
filled_length=$((bar_length * completed / total_files))
# Create the progress bar
bar=""
for ((i=0; i<bar_length; i++)); do
if [ $i -lt $filled_length ]; then
bar="${bar}█"
else
bar="${bar}░"
fi
done
# Calculate success and failure counts
success=$((completed - $(find "$compile_failures_dir" "$move_failures_dir" -type f | wc -l)))
compile_fails=$(find "$compile_failures_dir" -type f | wc -l)
move_fails=$(find "$move_failures_dir" -type f | wc -l)
# Clear the previous line and print the updated progress
echo -ne "\r\033[K"
echo -ne "${PURPLE}$ICON_WORKING Progress: ${GREEN}$bar ${BOLD}${percent}%${NC} "
echo -ne "[${GREEN}${success}✓${NC}|${RED}${compile_fails}✗${NC}|${YELLOW}${move_fails}!${NC}] "
echo -ne "${BLUE}($completed/$total_files)${NC}"
# If all files are processed, print the completion message ONLY ONCE
if [ $completed -eq $total_files ] && [ ! -e "$final_progress_file.done" ]; then
touch "$final_progress_file.done"
echo -e "\n${GREEN}${BOLD}$ICON_COMPLETE All files processed!${NC}"
fi
) 200>"$progress_lock"
}
# Function to show final progress (called only once at the end)
show_final_progress() {
# If progress is disabled, do nothing
if [ "$show_progress" -eq 0 ]; then
return 0
fi
(
flock -w 1 200
# Only proceed if the final progress hasn't been shown yet
if [ -e "$final_progress_file.done" ]; then
return 0
fi
completed=$(find "$progress_dir" -type f | wc -l)
percent=$((completed * 100 / total_files))
bar_length=50
filled_length=$((bar_length * completed / total_files))
# Create the progress bar
bar=""
for ((i=0; i<bar_length; i++)); do
if [ $i -lt $filled_length ]; then
bar="${bar}█"
else
bar="${bar}░"
fi
done
# Calculate success and failure counts
success=$((completed - $(find "$compile_failures_dir" "$move_failures_dir" -type f | wc -l)))
compile_fails=$(find "$compile_failures_dir" -type f | wc -l)
move_fails=$(find "$move_failures_dir" -type f | wc -l)
# Clear the previous line and print the updated progress
echo -ne "\r\033[K"
echo -ne "${PURPLE}$ICON_WORKING Progress: ${GREEN}$bar ${BOLD}${percent}%${NC} "
echo -ne "[${GREEN}${success}✓${NC}|${RED}${compile_fails}✗${NC}|${YELLOW}${move_fails}!${NC}] "
echo -ne "${BLUE}($completed/$total_files)${NC}"
# Mark final progress as shown
touch "$final_progress_file.done"
# If all files are processed, print the completion message
if [ $completed -eq $total_files ]; then
echo -e "\n${GREEN}${BOLD}$ICON_COMPLETE All files processed!${NC}"
fi
) 200>"$progress_lock"
}
# Function to process a single .typ file
process_file() {
typfile="$1"
file_id=$(echo "$typfile" | md5sum | cut -d' ' -f1)
# Get the directory containing the .typ file
typdir=$(dirname "$typfile")
# Get the filename without path
filename=$(basename "$typfile")
# Get the filename without extension
basename="${filename%.typ}"
# Check if output directory exists or should be created
target_dir="$typdir/$output_dir"
if [ ! -d "$target_dir" ]; then
if [ "$create_dirs" -eq 1 ]; then
if [ "$dry_run" -eq 0 ]; then
mkdir -p "$target_dir"
log_debug "Created directory: $target_dir"
else
log_debug "[DRY RUN] Would create directory: $target_dir"
fi
else
# Skip this file if output directory doesn't exist and --create-dirs not specified
log_debug "Skipping $typfile (no $output_dir directory)"
touch "$progress_dir/$file_id"
update_progress_during
return 0
fi
fi
# Skip if PDF is newer than source and --skip-newer is specified
if [ "$skip_newer" -eq 1 ] && [ -f "$target_dir/$basename.pdf" ]; then
if [ "$typfile" -ot "$target_dir/$basename.pdf" ] && [ "$force" -eq 0 ]; then
log_debug "Skipping $typfile (PDF is newer)"
touch "$progress_dir/$file_id"
update_progress_during
return 0
fi
fi
# Create a temporary file for capturing compiler output
temp_output="$temp_dir/output_${file_id}.log"
# Add a header to the log before compilation
{
echo -e "\n===== COMPILING: $typfile ====="
echo "$(date)"
} > "$temp_output.header"
# In dry run mode, just log what would be done
if [ "$dry_run" -eq 1 ]; then
log_debug "[DRY RUN] Would compile: $typfile"
log_debug "[DRY RUN] Would move to: $target_dir/$basename.pdf"
touch "$progress_dir/$file_id"
update_progress_during
return 0
fi
# Compile the .typ file using typst with --root flag and capture all output
if ! typst compile --root "$CWD" "$typfile" > "$temp_output.stdout" 2> "$temp_output.stderr"; then
# Store the failure
echo "$typfile" > "$compile_failures_dir/$file_id"
log_debug "Compilation failed for $typfile"
# Combine stdout and stderr
cat "$temp_output.stdout" "$temp_output.stderr" > "$temp_output.combined"
# Filter the output to only include error messages using ripgrep
rg "error:" -A 20 "$temp_output.combined" > "$temp_output.errors" || true
# Lock the log file to avoid concurrent writes corrupting it
(
flock -w 1 201
cat "$temp_output.header" "$temp_output.errors" >> "$compile_log"
echo -e "\n" >> "$compile_log"
) 201>"$compile_log.lock"
else
# Check if the output PDF exists
if [ -f "$typdir/$basename.pdf" ]; then
# Try to move the output PDF to the output directory
move_header="$temp_dir/move_${file_id}.header"
{
echo -e "\n===== MOVING: $typfile ====="
echo "$(date)"
} > "$move_header"
if ! mv "$typdir/$basename.pdf" "$target_dir/" 2> "$temp_output.move_err"; then
echo "$typfile -> $target_dir/$basename.pdf" > "$move_failures_dir/$file_id"
log_debug "Failed to move $basename.pdf to $target_dir/"
# Lock the log file to avoid concurrent writes corrupting it
(
flock -w 1 202
cat "$move_header" "$temp_output.move_err" >> "$move_log"
echo "Failed to move $typdir/$basename.pdf to $target_dir/" >> "$move_log"
) 202>"$move_log.lock"
else
log_debug "Moved $basename.pdf to $target_dir/"
fi
else
# This is a fallback check in case typst doesn't return error code
echo "$typfile" > "$compile_failures_dir/$file_id"
log_debug "Compilation completed without errors but no PDF was generated for $typfile"
# Lock the log file to avoid concurrent writes corrupting it
(
flock -w 1 201
echo "Compilation completed without errors but no PDF was generated" >> "$compile_log"
) 201>"$compile_log.lock"
fi
fi
# Mark this file as processed (for progress tracking)
touch "$progress_dir/$file_id"
# Update the progress bar
update_progress_during
}
export -f process_file
export -f update_progress_during
export -f show_final_progress
export -f log_debug
export -f log_info
export -f log_warning
export -f log_error
export -f log_success
export CWD
export temp_dir
export compile_failures_dir
export move_failures_dir
export progress_dir
export final_progress_file
export progress_lock
export compile_log
export move_log
export total_files
export GREEN BLUE YELLOW RED CYAN PURPLE NC BOLD
export ICON_SUCCESS ICON_ERROR ICON_WORKING ICON_COMPILE ICON_MOVE ICON_COMPLETE ICON_SUMMARY ICON_INFO ICON_DEBUG
export verbose
export quiet
export output_dir
export create_dirs
export dry_run
export skip_newer
export show_progress
export force
# Determine the number of CPU cores and use that many parallel jobs (if not specified)
if [ "$jobs" -eq 0 ]; then
jobs=$(nproc 2>/dev/null || sysctl -n hw.ncpu 2>/dev/null || echo 4)
fi
log_info "Using ${BOLD}$jobs${NC} parallel jobs for compilation"
# Initialize progress bar if showing progress
if [ "$show_progress" -eq 1 ]; then
update_progress_during
fi
# Process files in parallel with --will-cite to suppress citation notice
if [ "$total_files" -gt 0 ]; then
cat "$typst_files_list" | parallel --will-cite --jobs "$jobs" process_file
# Wait a moment for any remaining progress updates to complete
sleep 0.5
# Show the final progress exactly once if showing progress
if [ "$show_progress" -eq 1 ]; then
show_final_progress
fi
# Print summary of failures
if [ "$quiet" -eq 0 ]; then
echo -e "\n${BOLD}${PURPLE}$ICON_SUMMARY Processing Summary${NC}"
fi
# Collect all failure files
compile_failures=$(find "$compile_failures_dir" -type f | wc -l)
move_failures=$(find "$move_failures_dir" -type f | wc -l)
if [ "$compile_failures" -eq 0 ] && [ "$move_failures" -eq 0 ]; then
log_success "All files processed successfully."
else
if [ "$compile_failures" -gt 0 ]; then
echo -e "\n${RED}${BOLD}$ICON_ERROR Compilation failures:${NC}"
find "$compile_failures_dir" -type f -exec cat {} \; | sort | while read -r failure; do
echo -e "${RED}- $failure${NC}"
done
echo -e "${BLUE}See $compile_log for detailed error messages.${NC}"
fi
if [ "$move_failures" -gt 0 ]; then
echo -e "\n${YELLOW}${BOLD}$ICON_ERROR Move failures:${NC}"
find "$move_failures_dir" -type f -exec cat {} \; | sort | while read -r failure; do
echo -e "${YELLOW}- $failure${NC}"
done
echo -e "${BLUE}See $move_log for detailed error messages.${NC}"
fi
fi
else
log_warning "No .typ files found to process."
fi
# Clean up temporary directory and lock files
rm -rf "$temp_dir"
rm -f "$progress_lock" "$compile_log.lock" "$move_log.lock"
log_success "Processing complete."
1 | #!/usr/bin/env bash |
2 | |
3 | # Default values |
4 | jobs=0 # 0 means auto-detect |
5 | output_dir="pdfs" |
6 | verbose=0 |
7 | quiet=0 |
8 | create_dirs=0 |
9 | dry_run=0 |
10 | skip_newer=0 |
11 | show_progress=1 |
12 | force=0 |
13 | select_mode=0 |
14 | compile_log="typst_compile_errors.log" |
15 | move_log="typst_move_errors.log" |
16 | declare -a exclude_patterns |
17 | |
18 | # Show usage/help information |
19 | show_help() { |
20 | cat << EOF |
21 | Usage: $(basename "$0") [OPTIONS] |
22 | |
23 | Compile Typst files and move generated PDFs to a designated directory. |
24 | |
25 | Options: |
26 | -j, --jobs NUM Number of parallel jobs (default: auto-detect) |
27 | -o, --output-dir DIR Output directory name (default: pdfs) |
28 | -v, --verbose Increase verbosity |
29 | -q, --quiet Suppress most output |
30 | -c, --create-dirs Create output directories if they don't exist |
31 | -d, --dry-run Show what would be done without doing it |
32 | -s, --skip-newer Skip compilation if PDF is newer than source |
33 | -S, --select Interactive selection mode using skim |
34 | -f, --force Force compilation even if PDF exists |
35 | --no-progress Disable progress bar |
36 | --compile-log FILE Custom location for compilation log (default: $compile_log) |
37 | --move-log FILE Custom location for move log (default: $move_log) |
38 | -e, --exclude PATTERN Exclude files matching pattern (can be used multiple times) |
39 | -h, --help Show this help message and exit |
40 | |
41 | Examples: |
42 | $(basename "$0") -j 4 -o output -c |
43 | $(basename "$0") --verbose --skip-newer |
44 | $(basename "$0") -S # Select files to compile interactively |
45 | $(basename "$0") -e "**/test/**" -e "**/draft/**" |
46 | EOF |
47 | } |
48 | |
49 | # Parse command line arguments |
50 | while [[ $# -gt 0 ]]; do |
51 | case $1 in |
52 | -j|--jobs) |
53 | jobs="$2" |
54 | shift 2 |
55 | ;; |
56 | -o|--output-dir) |
57 | output_dir="$2" |
58 | shift 2 |
59 | ;; |
60 | -v|--verbose) |
61 | verbose=1 |
62 | shift |
63 | ;; |
64 | -q|--quiet) |
65 | quiet=1 |
66 | shift |
67 | ;; |
68 | -c|--create-dirs) |
69 | create_dirs=1 |
70 | shift |
71 | ;; |
72 | -d|--dry-run) |
73 | dry_run=1 |
74 | shift |
75 | ;; |
76 | -s|--skip-newer) |
77 | skip_newer=1 |
78 | shift |
79 | ;; |
80 | -S|--select) |
81 | select_mode=1 |
82 | shift |
83 | ;; |
84 | -f|--force) |
85 | force=1 |
86 | shift |
87 | ;; |
88 | --no-progress) |
89 | show_progress=0 |
90 | shift |
91 | ;; |
92 | --compile-log) |
93 | compile_log="$2" |
94 | shift 2 |
95 | ;; |
96 | --move-log) |
97 | move_log="$2" |
98 | shift 2 |
99 | ;; |
100 | -e|--exclude) |
101 | exclude_patterns+=("$2") |
102 | shift 2 |
103 | ;; |
104 | -h|--help) |
105 | show_help |
106 | exit 0 |
107 | ;; |
108 | *) |
109 | echo "Unknown option: $1" |
110 | show_help |
111 | exit 1 |
112 | ;; |
113 | esac |
114 | done |
115 | |
116 | # Check for conflicting options |
117 | if [ "$verbose" -eq 1 ] && [ "$quiet" -eq 1 ]; then |
118 | echo "Error: Cannot use both --verbose and --quiet" |
119 | exit 1 |
120 | fi |
121 | |
122 | # Check if required tools are installed |
123 | check_tool() { |
124 | if ! command -v "$1" &> /dev/null; then |
125 | echo "$1 is not installed. Please install it first." |
126 | echo "On most systems: $2" |
127 | exit 1 |
128 | fi |
129 | } |
130 | |
131 | check_tool "fd" "cargo install fd-find or apt/brew install fd-find" |
132 | check_tool "rg" "cargo install ripgrep or apt/brew install ripgrep" |
133 | check_tool "parallel" "apt/brew install parallel" |
134 | |
135 | # Check for skim and bat if select mode is enabled |
136 | if [ "$select_mode" -eq 1 ]; then |
137 | check_tool "sk" "cargo install skim or apt/brew install skim" |
138 | check_tool "bat" "cargo install bat or apt/brew install bat" |
139 | fi |
140 | |
141 | # ANSI color codes |
142 | GREEN='\033[0;32m' |
143 | BLUE='\033[0;34m' |
144 | YELLOW='\033[0;33m' |
145 | RED='\033[0;31m' |
146 | CYAN='\033[0;36m' |
147 | PURPLE='\033[0;35m' |
148 | NC='\033[0m' # No Color |
149 | BOLD='\033[1m' |
150 | |
151 | # Nerd font icons |
152 | ICON_SUCCESS=" " |
153 | ICON_ERROR=" " |
154 | ICON_WORKING=" " |
155 | ICON_COMPILE=" " |
156 | ICON_MOVE=" " |
157 | ICON_COMPLETE=" " |
158 | ICON_SUMMARY=" " |
159 | ICON_INFO=" " |
160 | ICON_DEBUG=" " |
161 | ICON_SELECT=" " |
162 | |
163 | # Logging functions |
164 | log_debug() { |
165 | if [ "$verbose" -eq 1 ]; then |
166 | echo -e "${BLUE}$ICON_DEBUG${NC} $*" |
167 | fi |
168 | } |
169 | |
170 | log_info() { |
171 | if [ "$quiet" -eq 0 ]; then |
172 | echo -e "${CYAN}$ICON_INFO${NC} $*" |
173 | fi |
174 | } |
175 | |
176 | log_warning() { |
177 | if [ "$quiet" -eq 0 ]; then |
178 | echo -e "${YELLOW}⚠️ ${NC} $*" |
179 | fi |
180 | } |
181 | |
182 | log_error() { |
183 | echo -e "${RED}${BOLD}$ICON_ERROR${NC} $*" |
184 | } |
185 | |
186 | log_success() { |
187 | if [ "$quiet" -eq 0 ]; then |
188 | echo -e "${GREEN}${BOLD}$ICON_SUCCESS${NC} $*" |
189 | fi |
190 | } |
191 | |
192 | # Create a directory for temporary files |
193 | temp_dir=$(mktemp -d) |
194 | compile_failures_dir="$temp_dir/compile_failures" |
195 | move_failures_dir="$temp_dir/move_failures" |
196 | progress_dir="$temp_dir/progress" |
197 | mkdir -p "$compile_failures_dir" "$move_failures_dir" "$progress_dir" |
198 | |
199 | # Create a lock file for progress updates and a flag for final progress |
200 | progress_lock="/tmp/typst_progress_lock" |
201 | final_progress_file="$temp_dir/final_progress" |
202 | touch "$final_progress_file" |
203 | |
204 | # Initialize log files |
205 | > "$compile_log" |
206 | > "$move_log" |
207 | |
208 | # Store current working directory |
209 | CWD=$(pwd) |
210 | |
211 | log_info "Starting Typst compilation process..." |
212 | |
213 | # Build exclude arguments for fd |
214 | fd_exclude_args=() |
215 | for pattern in "${exclude_patterns[@]}"; do |
216 | fd_exclude_args+=("-E" "$pattern") |
217 | done |
218 | |
219 | # Create a list of files to process |
220 | typst_files_list="$temp_dir/typst_files.txt" |
221 | |
222 | # Create a custom bat configuration for Typst syntax |
223 | setup_bat_for_typst() { |
224 | bat_config_dir="$temp_dir/bat_config" |
225 | mkdir -p "$bat_config_dir/syntaxes" |
226 | |
227 | # Create a basic syntax mapping file for Typst |
228 | cat > "$bat_config_dir/syntaxes/typst.sublime-syntax" << 'TYPST_SYNTAX' |
229 | %YAML 1.2 |
230 | --- |
231 | name: Typst |
232 | file_extensions: |
233 | - typ |
234 | scope: source.typst |
235 | |
236 | contexts: |
237 | main: |
238 | # Comments |
239 | - match: /\* |
240 | scope: comment.block.typst |
241 | push: block_comment |
242 | |
243 | - match: // |
244 | scope: comment.line.double-slash.typst |
245 | push: line_comment |
246 | |
247 | # Strings |
248 | - match: '"' |
249 | scope: punctuation.definition.string.begin.typst |
250 | push: double_string |
251 | |
252 | # Math |
253 | - match: '\$' |
254 | scope: punctuation.definition.math.begin.typst |
255 | push: math |
256 | |
257 | # Functions |
258 | - match: '#([a-zA-Z][a-zA-Z0-9_]*)' |
259 | scope: entity.name.function.typst |
260 | |
261 | # Variables |
262 | - match: '\b([a-zA-Z][a-zA-Z0-9_]*)\s*:' |
263 | scope: variable.other.typst |
264 | |
265 | # Keywords |
266 | - match: '\b(let|set|show|if|else|for|in|while|return|import|include|at|do|not|and|or|none|auto)\b' |
267 | scope: keyword.control.typst |
268 | |
269 | block_comment: |
270 | - match: \*/ |
271 | scope: comment.block.typst |
272 | pop: true |
273 | - match: . |
274 | scope: comment.block.typst |
275 | |
276 | line_comment: |
277 | - match: $ |
278 | pop: true |
279 | - match: . |
280 | scope: comment.line.double-slash.typst |
281 | |
282 | double_string: |
283 | - match: '"' |
284 | scope: punctuation.definition.string.end.typst |
285 | pop: true |
286 | - match: \\. |
287 | scope: constant.character.escape.typst |
288 | - match: . |
289 | scope: string.quoted.double.typst |
290 | |
291 | math: |
292 | - match: '\$' |
293 | scope: punctuation.definition.math.end.typst |
294 | pop: true |
295 | - match: . |
296 | scope: markup.math.typst |
297 | TYPST_SYNTAX |
298 | |
299 | echo "$bat_config_dir" |
300 | } |
301 | |
302 | if [ "$select_mode" -eq 1 ]; then |
303 | log_info "${PURPLE}$ICON_SELECT${NC} Interactive selection mode enabled" |
304 | log_info "Use TAB to select multiple files, ENTER to confirm" |
305 | |
306 | # Set up bat configuration for Typst syntax highlighting |
307 | bat_config=$(setup_bat_for_typst) |
308 | export BAT_CONFIG_PATH="$bat_config" |
309 | |
310 | # Use skim with bat for preview to select files |
311 | selected_files="$temp_dir/selected_files.txt" |
312 | |
313 | # Find all eligible .typ files for selection |
314 | fd '\.typ$' --type f "${fd_exclude_args[@]}" . > "$typst_files_list.all" |
315 | |
316 | # Check if we found any files |
317 | if [ ! -s "$typst_files_list.all" ]; then |
318 | log_error "No .typ files found matching your criteria." |
319 | rm -rf "$temp_dir" |
320 | exit 0 |
321 | fi |
322 | |
323 | # Prepare preview command for skim: use bat with custom syntax for .typ files |
324 | preview_cmd="bat --color=always --style=numbers --map-syntax='*.typ:Markdown' {} 2>/dev/null || cat {}" |
325 | |
326 | # Use skim with bat preview to select files |
327 | cat "$typst_files_list.all" | sk --multi \ |
328 | --preview "$preview_cmd" \ |
329 | --preview-window "right:70%" \ |
330 | --height "80%" \ |
331 | --prompt "Select .typ files to compile: " \ |
332 | --header "TAB: Select multiple files, ENTER: Confirm, CTRL-C: Cancel" \ |
333 | --no-mouse > "$selected_files" |
334 | |
335 | # Check if user selected any files |
336 | if [ ! -s "$selected_files" ]; then |
337 | log_error "No files selected. Exiting." |
338 | rm -rf "$temp_dir" |
339 | exit 0 |
340 | fi |
341 | |
342 | # Use the selected files instead of all discovered files |
343 | cp "$selected_files" "$typst_files_list" |
344 | total_selected=$(wc -l < "$typst_files_list") |
345 | total_available=$(wc -l < "$typst_files_list.all") |
346 | |
347 | log_info "Selected ${BOLD}$total_selected${NC} out of ${BOLD}$total_available${NC} available files" |
348 | |
349 | # Display tip about better Typst syntax highlighting |
350 | cat << 'TYPST_TIP' |
351 | |
352 | 📝 To get proper Typst syntax highlighting in bat: |
353 | |
354 | 1. Create a custom Typst syntax file: |
355 | mkdir -p ~/.config/bat/syntaxes |
356 | curl -L https://raw.githubusercontent.com/typst/typst-vs-code/main/syntaxes/typst.tmLanguage.json \ |
357 | -o ~/.config/bat/syntaxes/typst.tmLanguage.json |
358 | |
359 | 2. Build bat's syntax cache: |
360 | bat cache --build |
361 | |
362 | This will provide proper syntax highlighting for Typst files in future uses! |
363 | |
364 | TYPST_TIP |
365 | else |
366 | # Normal mode - process all files |
367 | fd '\.typ$' --type f "${fd_exclude_args[@]}" . > "$typst_files_list" |
368 | log_info "Found ${BOLD}$(wc -l < "$typst_files_list")${NC} Typst files to process" |
369 | fi |
370 | |
371 | total_files=$(wc -l < "$typst_files_list") |
372 | |
373 | # Function to update progress bar during processing |
374 | update_progress_during() { |
375 | # If progress is disabled, do nothing |
376 | if [ "$show_progress" -eq 0 ]; then |
377 | return 0 |
378 | fi |
379 | |
380 | ( |
381 | # Try to acquire lock, but don't wait if busy |
382 | flock -n 200 || return 0 |
383 | |
384 | completed=$(find "$progress_dir" -type f | wc -l) |
385 | percent=$((completed * 100 / total_files)) |
386 | bar_length=50 |
387 | filled_length=$((bar_length * completed / total_files)) |
388 | |
389 | # Create the progress bar |
390 | bar="" |
391 | for ((i=0; i<bar_length; i++)); do |
392 | if [ $i -lt $filled_length ]; then |
393 | bar="${bar}█" |
394 | else |
395 | bar="${bar}░" |
396 | fi |
397 | done |
398 | |
399 | # Calculate success and failure counts |
400 | success=$((completed - $(find "$compile_failures_dir" "$move_failures_dir" -type f | wc -l))) |
401 | compile_fails=$(find "$compile_failures_dir" -type f | wc -l) |
402 | move_fails=$(find "$move_failures_dir" -type f | wc -l) |
403 | |
404 | # Clear the previous line and print the updated progress |
405 | echo -ne "\r\033[K" |
406 | echo -ne "${PURPLE}$ICON_WORKING Progress: ${GREEN}$bar ${BOLD}${percent}%${NC} " |
407 | echo -ne "[${GREEN}${success}✓${NC}|${RED}${compile_fails}✗${NC}|${YELLOW}${move_fails}!${NC}] " |
408 | echo -ne "${BLUE}($completed/$total_files)${NC}" |
409 | |
410 | # If all files are processed, print the completion message ONLY ONCE |
411 | if [ $completed -eq $total_files ] && [ ! -e "$final_progress_file.done" ]; then |
412 | touch "$final_progress_file.done" |
413 | echo -e "\n${GREEN}${BOLD}$ICON_COMPLETE All files processed!${NC}" |
414 | fi |
415 | ) 200>"$progress_lock" |
416 | } |
417 | |
418 | # Function to show final progress (called only once at the end) |
419 | show_final_progress() { |
420 | # If progress is disabled, do nothing |
421 | if [ "$show_progress" -eq 0 ]; then |
422 | return 0 |
423 | fi |
424 | |
425 | ( |
426 | flock -w 1 200 |
427 | |
428 | # Only proceed if the final progress hasn't been shown yet |
429 | if [ -e "$final_progress_file.done" ]; then |
430 | return 0 |
431 | fi |
432 | |
433 | completed=$(find "$progress_dir" -type f | wc -l) |
434 | percent=$((completed * 100 / total_files)) |
435 | bar_length=50 |
436 | filled_length=$((bar_length * completed / total_files)) |
437 | |
438 | # Create the progress bar |
439 | bar="" |
440 | for ((i=0; i<bar_length; i++)); do |
441 | if [ $i -lt $filled_length ]; then |
442 | bar="${bar}█" |
443 | else |
444 | bar="${bar}░" |
445 | fi |
446 | done |
447 | |
448 | # Calculate success and failure counts |
449 | success=$((completed - $(find "$compile_failures_dir" "$move_failures_dir" -type f | wc -l))) |
450 | compile_fails=$(find "$compile_failures_dir" -type f | wc -l) |
451 | move_fails=$(find "$move_failures_dir" -type f | wc -l) |
452 | |
453 | # Clear the previous line and print the updated progress |
454 | echo -ne "\r\033[K" |
455 | echo -ne "${PURPLE}$ICON_WORKING Progress: ${GREEN}$bar ${BOLD}${percent}%${NC} " |
456 | echo -ne "[${GREEN}${success}✓${NC}|${RED}${compile_fails}✗${NC}|${YELLOW}${move_fails}!${NC}] " |
457 | echo -ne "${BLUE}($completed/$total_files)${NC}" |
458 | |
459 | # Mark final progress as shown |
460 | touch "$final_progress_file.done" |
461 | |
462 | # If all files are processed, print the completion message |
463 | if [ $completed -eq $total_files ]; then |
464 | echo -e "\n${GREEN}${BOLD}$ICON_COMPLETE All files processed!${NC}" |
465 | fi |
466 | ) 200>"$progress_lock" |
467 | } |
468 | |
469 | # Function to process a single .typ file |
470 | process_file() { |
471 | typfile="$1" |
472 | file_id=$(echo "$typfile" | md5sum | cut -d' ' -f1) |
473 | |
474 | # Get the directory containing the .typ file |
475 | typdir=$(dirname "$typfile") |
476 | # Get the filename without path |
477 | filename=$(basename "$typfile") |
478 | # Get the filename without extension |
479 | basename="${filename%.typ}" |
480 | |
481 | # Check if output directory exists or should be created |
482 | target_dir="$typdir/$output_dir" |
483 | if [ ! -d "$target_dir" ]; then |
484 | if [ "$create_dirs" -eq 1 ]; then |
485 | if [ "$dry_run" -eq 0 ]; then |
486 | mkdir -p "$target_dir" |
487 | log_debug "Created directory: $target_dir" |
488 | else |
489 | log_debug "[DRY RUN] Would create directory: $target_dir" |
490 | fi |
491 | else |
492 | # Skip this file if output directory doesn't exist and --create-dirs not specified |
493 | log_debug "Skipping $typfile (no $output_dir directory)" |
494 | touch "$progress_dir/$file_id" |
495 | update_progress_during |
496 | return 0 |
497 | fi |
498 | fi |
499 | |
500 | # Skip if PDF is newer than source and --skip-newer is specified |
501 | if [ "$skip_newer" -eq 1 ] && [ -f "$target_dir/$basename.pdf" ]; then |
502 | if [ "$typfile" -ot "$target_dir/$basename.pdf" ] && [ "$force" -eq 0 ]; then |
503 | log_debug "Skipping $typfile (PDF is newer)" |
504 | touch "$progress_dir/$file_id" |
505 | update_progress_during |
506 | return 0 |
507 | fi |
508 | fi |
509 | |
510 | # Create a temporary file for capturing compiler output |
511 | temp_output="$temp_dir/output_${file_id}.log" |
512 | |
513 | # Add a header to the log before compilation |
514 | { |
515 | echo -e "\n===== COMPILING: $typfile =====" |
516 | echo "$(date)" |
517 | } > "$temp_output.header" |
518 | |
519 | # In dry run mode, just log what would be done |
520 | if [ "$dry_run" -eq 1 ]; then |
521 | log_debug "[DRY RUN] Would compile: $typfile" |
522 | log_debug "[DRY RUN] Would move to: $target_dir/$basename.pdf" |
523 | touch "$progress_dir/$file_id" |
524 | update_progress_during |
525 | return 0 |
526 | fi |
527 | |
528 | # Compile the .typ file using typst with --root flag and capture all output |
529 | if ! typst compile --root "$CWD" "$typfile" > "$temp_output.stdout" 2> "$temp_output.stderr"; then |
530 | # Store the failure |
531 | echo "$typfile" > "$compile_failures_dir/$file_id" |
532 | log_debug "Compilation failed for $typfile" |
533 | |
534 | # Combine stdout and stderr |
535 | cat "$temp_output.stdout" "$temp_output.stderr" > "$temp_output.combined" |
536 | |
537 | # Filter the output to only include error messages using ripgrep |
538 | rg "error:" -A 20 "$temp_output.combined" > "$temp_output.errors" || true |
539 | |
540 | # Lock the log file to avoid concurrent writes corrupting it |
541 | ( |
542 | flock -w 1 201 |
543 | cat "$temp_output.header" "$temp_output.errors" >> "$compile_log" |
544 | echo -e "\n" >> "$compile_log" |
545 | ) 201>"$compile_log.lock" |
546 | else |
547 | # Check if the output PDF exists |
548 | if [ -f "$typdir/$basename.pdf" ]; then |
549 | # Try to move the output PDF to the output directory |
550 | move_header="$temp_dir/move_${file_id}.header" |
551 | { |
552 | echo -e "\n===== MOVING: $typfile =====" |
553 | echo "$(date)" |
554 | } > "$move_header" |
555 | |
556 | if ! mv "$typdir/$basename.pdf" "$target_dir/" 2> "$temp_output.move_err"; then |
557 | echo "$typfile -> $target_dir/$basename.pdf" > "$move_failures_dir/$file_id" |
558 | log_debug "Failed to move $basename.pdf to $target_dir/" |
559 | |
560 | # Lock the log file to avoid concurrent writes corrupting it |
561 | ( |
562 | flock -w 1 202 |
563 | cat "$move_header" "$temp_output.move_err" >> "$move_log" |
564 | echo "Failed to move $typdir/$basename.pdf to $target_dir/" >> "$move_log" |
565 | ) 202>"$move_log.lock" |
566 | else |
567 | log_debug "Moved $basename.pdf to $target_dir/" |
568 | fi |
569 | else |
570 | # This is a fallback check in case typst doesn't return error code |
571 | echo "$typfile" > "$compile_failures_dir/$file_id" |
572 | log_debug "Compilation completed without errors but no PDF was generated for $typfile" |
573 | |
574 | # Lock the log file to avoid concurrent writes corrupting it |
575 | ( |
576 | flock -w 1 201 |
577 | echo "Compilation completed without errors but no PDF was generated" >> "$compile_log" |
578 | ) 201>"$compile_log.lock" |
579 | fi |
580 | fi |
581 | |
582 | # Mark this file as processed (for progress tracking) |
583 | touch "$progress_dir/$file_id" |
584 | |
585 | # Update the progress bar |
586 | update_progress_during |
587 | } |
588 | |
589 | export -f process_file |
590 | export -f update_progress_during |
591 | export -f show_final_progress |
592 | export -f log_debug |
593 | export -f log_info |
594 | export -f log_warning |
595 | export -f log_error |
596 | export -f log_success |
597 | export CWD |
598 | export temp_dir |
599 | export compile_failures_dir |
600 | export move_failures_dir |
601 | export progress_dir |
602 | export final_progress_file |
603 | export progress_lock |
604 | export compile_log |
605 | export move_log |
606 | export total_files |
607 | export GREEN BLUE YELLOW RED CYAN PURPLE NC BOLD |
608 | export ICON_SUCCESS ICON_ERROR ICON_WORKING ICON_COMPILE ICON_MOVE ICON_COMPLETE ICON_SUMMARY ICON_INFO ICON_DEBUG |
609 | export verbose |
610 | export quiet |
611 | export output_dir |
612 | export create_dirs |
613 | export dry_run |
614 | export skip_newer |
615 | export show_progress |
616 | export force |
617 | |
618 | # Determine the number of CPU cores and use that many parallel jobs (if not specified) |
619 | if [ "$jobs" -eq 0 ]; then |
620 | jobs=$(nproc 2>/dev/null || sysctl -n hw.ncpu 2>/dev/null || echo 4) |
621 | fi |
622 | log_info "Using ${BOLD}$jobs${NC} parallel jobs for compilation" |
623 | |
624 | # Initialize progress bar if showing progress |
625 | if [ "$show_progress" -eq 1 ]; then |
626 | update_progress_during |
627 | fi |
628 | |
629 | # Process files in parallel with --will-cite to suppress citation notice |
630 | if [ "$total_files" -gt 0 ]; then |
631 | cat "$typst_files_list" | parallel --will-cite --jobs "$jobs" process_file |
632 | |
633 | # Wait a moment for any remaining progress updates to complete |
634 | sleep 0.5 |
635 | |
636 | # Show the final progress exactly once if showing progress |
637 | if [ "$show_progress" -eq 1 ]; then |
638 | show_final_progress |
639 | fi |
640 | |
641 | # Print summary of failures |
642 | if [ "$quiet" -eq 0 ]; then |
643 | echo -e "\n${BOLD}${PURPLE}$ICON_SUMMARY Processing Summary${NC}" |
644 | fi |
645 | |
646 | # Collect all failure files |
647 | compile_failures=$(find "$compile_failures_dir" -type f | wc -l) |
648 | move_failures=$(find "$move_failures_dir" -type f | wc -l) |
649 | |
650 | if [ "$compile_failures" -eq 0 ] && [ "$move_failures" -eq 0 ]; then |
651 | log_success "All files processed successfully." |
652 | else |
653 | if [ "$compile_failures" -gt 0 ]; then |
654 | echo -e "\n${RED}${BOLD}$ICON_ERROR Compilation failures:${NC}" |
655 | find "$compile_failures_dir" -type f -exec cat {} \; | sort | while read -r failure; do |
656 | echo -e "${RED}- $failure${NC}" |
657 | done |
658 | echo -e "${BLUE}See $compile_log for detailed error messages.${NC}" |
659 | fi |
660 | |
661 | if [ "$move_failures" -gt 0 ]; then |
662 | echo -e "\n${YELLOW}${BOLD}$ICON_ERROR Move failures:${NC}" |
663 | find "$move_failures_dir" -type f -exec cat {} \; | sort | while read -r failure; do |
664 | echo -e "${YELLOW}- $failure${NC}" |
665 | done |
666 | echo -e "${BLUE}See $move_log for detailed error messages.${NC}" |
667 | fi |
668 | fi |
669 | else |
670 | log_warning "No .typ files found to process." |
671 | fi |
672 | |
673 | # Clean up temporary directory and lock files |
674 | rm -rf "$temp_dir" |
675 | rm -f "$progress_lock" "$compile_log.lock" "$move_log.lock" |
676 | |
677 | log_success "Processing complete." |
678 |