compile_typst.sh
· 33 KiB · Bash
Неформатований
#!/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
use_cache=1 # Enable caching by default
use_git=1 # Enable Git integration by default
cache_dir="${XDG_CACHE_HOME:-$HOME/.cache}/typst_compiler"
dependency_scan=1 # Enable dependency scanning by default
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
--no-cache Disable compilation caching
--no-git Disable Git-based change detection
--clear-cache Clear the cache before compiling
--no-deps Disable dependency scanning
--cache-dir DIR Custom cache directory (default: ~/.cache/typst_compiler)
--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/**"
$(basename "$0") --no-git # Disable Git-based change detection
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
;;
--no-cache)
use_cache=0
shift
;;
--no-git)
use_git=0
shift
;;
--clear-cache)
rm -rf "${cache_dir}"
shift
;;
--no-deps)
dependency_scan=0
shift
;;
--cache-dir)
cache_dir="$2"
shift 2
;;
--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_tool "sha256sum" "Built-in on most Linux systems, on macOS: brew install coreutils"
# 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=" "
ICON_CACHE=" "
ICON_GIT=" "
# 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 cache directories if caching is enabled
if [ "$use_cache" -eq 1 ]; then
mkdir -p "${cache_dir}/hashes" "${cache_dir}/deps" "${cache_dir}/pdfs"
log_debug "Using cache directory: $cache_dir"
fi
# 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..."
# Check if we're in a Git repository
in_git_repo=0
git_repo_root=""
check_git_repo() {
if [ "$use_git" -eq 1 ] && command -v git &> /dev/null; then
# Check if we're in a git repo
if git_repo_root=$(git rev-parse --show-toplevel 2>/dev/null); then
in_git_repo=1
log_info "${ICON_GIT} Git repository detected at: ${BOLD}$git_repo_root${NC}"
return 0
else
log_debug "Not in a Git repository, falling back to file hashing"
return 1
fi
else
log_debug "Git integration disabled or git not found"
return 1
fi
}
# Get list of changed files from Git
get_git_changed_files() {
local git_files="$temp_dir/git_changed_files.txt"
# Get list of staged, unstaged, and untracked files
git -C "$git_repo_root" ls-files --modified --others --exclude-standard > "$git_files"
# Add staged files that may not show as modified
git -C "$git_repo_root" diff --name-only --cached >> "$git_files"
# If no previous commit exists, consider all tracked files as changed
if ! git -C "$git_repo_root" rev-parse HEAD &>/dev/null; then
git -C "$git_repo_root" ls-files >> "$git_files"
fi
# Filter to keep only .typ files and remove duplicates
sort -u "$git_files" | grep '\.typ$' > "$temp_dir/git_changed_typ_files.txt"
log_debug "Git reports $(wc -l < "$temp_dir/git_changed_typ_files.txt") changed .typ files"
}
# 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"
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="$temp_dir/bat_config"
mkdir -p "$bat_config/syntaxes"
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"
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"
# Check for Git repository and get changed files
check_git_repo
if [ "$in_git_repo" -eq 1 ] && [ "$force" -eq 0 ]; then
get_git_changed_files
# If no forced compilation, filter the file list to only include changed files
if [ -s "$temp_dir/git_changed_typ_files.txt" ]; then
cp "$typst_files_list" "$temp_dir/all_typ_files.txt"
# Convert Git changed files to absolute paths
while read -r file; do
echo "$git_repo_root/$file"
done < "$temp_dir/git_changed_typ_files.txt" | sort > "$temp_dir/git_absolute_paths.txt"
# Find the intersection of all files and Git changed files
comm -12 <(sort "$typst_files_list") "$temp_dir/git_absolute_paths.txt" > "$temp_dir/files_to_compile.txt"
# Use the filtered list if there are changes, otherwise keep using all files
if [ -s "$temp_dir/files_to_compile.txt" ]; then
cp "$temp_dir/files_to_compile.txt" "$typst_files_list"
git_changes_count=$(wc -l < "$typst_files_list")
log_info "${ICON_GIT} Processing ${BOLD}$git_changes_count${NC} files with Git changes"
else
log_info "${ICON_GIT} No Git changes detected for .typ files"
# If no changes detected and not forced, update progress and exit early
if [ "$force" -eq 0 ]; then
log_success "No files need to be compiled."
rm -rf "$temp_dir"
exit 0
fi
fi
fi
fi
fi
total_files=$(wc -l < "$typst_files_list")
# Function to extract Typst imports from a file
extract_dependencies() {
local file="$1"
# Match both import formats: #import "file.typ" and #include "file.typ"
# Also handle package imports: #import "@preview:0.1.0"
# Return full paths to local imports, skipping package imports
rg -U "#(import|include) \"(?!@)([^\"]+)\"" -r "$CWD/\$2" "$file" | \
while read -r dep; do
# Resolve relative paths correctly
if [[ "$dep" != /* ]]; then
# If not absolute path, make it relative to the file's directory
file_dir=$(dirname "$file")
dep="$file_dir/$dep"
fi
# Normalize path
dep=$(realpath --relative-to="$CWD" "$dep" 2>/dev/null || echo "$dep")
# Only output if the dependency exists and is a .typ file
if [[ "$dep" == *.typ ]] && [ -f "$dep" ]; then
echo "$dep"
fi
done | sort -u # Sort and remove duplicates
}
# Function to build dependency graph for all files
build_dependency_graph() {
log_info "${ICON_CACHE} Building dependency graph..."
# Create a deps file for each Typst file
while read -r file; do
file_hash=$(echo "$file" | md5sum | cut -d' ' -f1)
deps_file="${cache_dir}/deps/${file_hash}.deps"
# Skip if deps file is fresh and not forced
if [ "$force" -eq 0 ] && [ -f "$deps_file" ] && [ "$file" -ot "$deps_file" ]; then
log_debug "Using cached dependencies for $file"
continue
fi
# Extract dependencies and save to deps file
extract_dependencies "$file" > "$deps_file"
log_debug "Updated dependencies for $file"
done < "$typst_files_list"
# Create sorted compilation order based on dependencies
# Files with fewer dependencies (or no dependencies) come first
if [ -f "$temp_dir/compile_order.txt" ]; then
rm "$temp_dir/compile_order.txt"
fi
# First pass: count dependencies for each file
declare -A dep_counts
while read -r file; do
file_hash=$(echo "$file" | md5sum | cut -d' ' -f1)
deps_file="${cache_dir}/deps/${file_hash}.deps"
# Count dependencies
if [ -f "$deps_file" ]; then
dep_count=$(wc -l < "$deps_file")
else
dep_count=0
fi
dep_counts["$file"]=$dep_count
done < "$typst_files_list"
# Second pass: sort files by dependency count (ascending)
while read -r file; do
echo "${dep_counts["$file"]} $file"
done < "$typst_files_list" | sort -n | cut -d' ' -f2- > "$temp_dir/compile_order.txt"
log_debug "Created optimized compilation order"
# If in a Git repo, consider dependent files of changed files
if [ "$in_git_repo" -eq 1 ] && [ -f "$temp_dir/git_changed_typ_files.txt" ]; then
log_info "${ICON_GIT} Analyzing dependencies of changed files..."
# Track files to be compiled (start with directly changed files)
cp "$typst_files_list" "$temp_dir/files_to_compile_with_deps.txt"
# Identify files that depend on changed files (reverse dependencies)
while read -r file; do
file_hash=$(echo "$file" | md5sum | cut -d' ' -f1)
# Find all files that have this file as a dependency
grep -l "^$file$" "${cache_dir}"/deps/*.deps 2>/dev/null | while read -r dep_file; do
# Extract the dependent file from the hash
dependent_hash=$(basename "$dep_file" .deps)
# Find the original filename for this hash
while read -r potential_file; do
potential_hash=$(echo "$potential_file" | md5sum | cut -d' ' -f1)
if [ "$potential_hash" == "$dependent_hash" ]; then
# Add this dependent file to the compilation list
echo "$potential_file" >> "$temp_dir/files_to_compile_with_deps.txt"
break
fi
done < "$temp_dir/all_typ_files.txt"
done
done < "$typst_files_list"
# Remove duplicates and update the files list
sort -u "$temp_dir/files_to_compile_with_deps.txt" > "$typst_files_list"
# Update the total count
new_total=$(wc -l < "$typst_files_list")
if [ "$new_total" -gt "$total_files" ]; then
log_info "${ICON_GIT} Added ${BOLD}$((new_total - total_files))${NC} additional files with dependencies"
total_files=$new_total
fi
fi
}
# Function to check if a file has changed since last compilation
has_file_changed() {
local file="$1"
# Force recompilation if requested
if [ "$force" -eq 1 ]; then
log_debug "Force recompilation of $file"
return 0 # File considered changed
fi
# If using Git and in a Git repo, check if file is in the changed list
if [ "$in_git_repo" -eq 1 ] && [ "$use_git" -eq 1 ]; then
# Convert to repo-relative path
local repo_path="${file#$git_repo_root/}"
# Check if this file or any of its dependencies are changed according to Git
if grep -q "^$git_repo_root/$repo_path$" "$temp_dir/git_absolute_paths.txt" 2>/dev/null; then
log_debug "Git reports $file has changed"
return 0 # File considered changed
fi
# If not in the changed list and we're only processing Git changes,
# then this file must have a dependency that changed
if [ -f "$temp_dir/git_changed_typ_files.txt" ]; then
# Check dependencies (if enabled)
if [ "$dependency_scan" -eq 1 ]; then
local file_hash=$(echo "$file" | md5sum | cut -d' ' -f1)
local deps_file="${cache_dir}/deps/${file_hash}.deps"
if [ -f "$deps_file" ]; then
while read -r dep; do
if has_file_changed "$dep"; then
log_debug "Dependency $dep of $file has changed"
return 0 # Dependency has changed
fi
done < "$deps_file"
fi
fi
log_debug "No changes to $file or its dependencies according to Git"
return 1 # No changes detected via Git
fi
fi
# Fall back to hash-based detection if Git not available or not conclusive
local file_hash=$(echo "$file" | md5sum | cut -d' ' -f1)
local hash_file="${cache_dir}/hashes/${file_hash}.sha256"
# Check if hash file exists
if [ ! -f "$hash_file" ]; then
log_debug "No previous hash for $file"
return 0 # File considered changed
fi
# Compare current file hash with stored hash
local current_hash=$(sha256sum "$file" | cut -d' ' -f1)
local stored_hash=$(cat "$hash_file")
if [ "$current_hash" != "$stored_hash" ]; then
log_debug "File $file has changed since last compilation"
return 0 # File has changed
fi
# Check if any dependencies have changed
if [ "$dependency_scan" -eq 1 ]; then
local deps_file="${cache_dir}/deps/${file_hash}.deps"
if [ -f "$deps_file" ]; then
while read -r dep; do
if has_file_changed "$dep"; then
log_debug "Dependency $dep of $file has changed"
return 0 # Dependency has changed
fi
done < "$deps_file"
fi
fi
log_debug "File $file and its dependencies are unchanged"
return 1 # File and dependencies haven't changed
}
# Function to update file hash after compilation
update_file_hash() {
local file="$1"
local file_hash=$(echo "$file" | md5sum | cut -d' ' -f1)
local hash_file="${cache_dir}/hashes/${file_hash}.sha256"
# Update hash file
sha256sum "$file" | cut -d' ' -f1 > "$hash_file"
log_debug "Updated hash for $file"
}
# Function to cache compiled PDF
cache_pdf() {
local src_file="$1"
local pdf_file="$2"
if [ ! -f "$pdf_file" ]; then
log_debug "No PDF file to cache: $pdf_file"
return 1
fi
local file_hash=$(echo "$src_file" | md5sum | cut -d' ' -f1)
local cached_pdf="${cache_dir}/pdfs/${file_hash}.pdf"
# Copy PDF to cache
cp "$pdf_file" "$cached_pdf"
log_debug "Cached PDF for $src_file"
return 0
}
# Function to retrieve cached PDF
get_cached_pdf() {
local src_file="$1"
local target_file="$2"
local file_hash=$(echo "$src_file" | md5sum | cut -d' ' -f1)
local cached_pdf="${cache_dir}/pdfs/${file_hash}.pdf"
if [ ! -f "$cached_pdf" ]; then
log_debug "No cached PDF for $src_file"
return 1
fi
# Copy cached PDF to target location
cp "$cached_pdf" "$target_file"
log_debug "Retrieved cached PDF for $src_file"
return 0
}
# 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
target_pdf="$target_dir/$basename.pdf"
# Skip if PDF is newer than source and --skip-newer is specified
if [ "$skip_newer" -eq 1 ] && [ -f "$target_pdf" ]; then
if [ "$typfile" -ot "$target_pdf" ] && [ "$force" -eq 0 ]; then
log_debug "Skipping $typfile (PDF is newer)"
touch "$progress_dir/$file_id"
update_progress_during
return 0
fi
fi
# Check if file has changed (if caching is enabled)
if [ "$use_cache" -eq 1 ] && [ "$dry_run" -eq 0 ]; then
if ! has_file_changed "$typfile"; then
# Try to retrieve cached PDF
if get_cached_pdf "$typfile" "$target_pdf"; then
log_debug "Using cached PDF for $typfile"
touch "$progress_dir/$file_id"
update_progress_during
return 0
fi
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_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
temp_pdf="$typdir/$basename.pdf"
if [ -f "$temp_pdf" ]; then
# Cache the PDF if caching is enabled
if [ "$use_cache" -eq 1 ]; then
cache_pdf "$typfile" "$temp_pdf"
# Update file hash
update_file_hash "$typfile"
fi
# 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 "$temp_pdf" "$target_dir/" 2> "$temp_output.move_err"; then
echo "$typfile -> $target_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 $temp_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 -f has_file_changed
export -f update_file_hash
export -f cache_pdf
export -f get_cached_pdf
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 cache_dir
export git_repo_root
export in_git_repo
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 ICON_CACHE ICON_GIT
export verbose
export quiet
export output_dir
export create_dirs
export dry_run
export skip_newer
export show_progress
export force
export use_cache
export dependency_scan
export use_git
# 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"
# Build dependency graph if dependency scanning is enabled
if [ "$dependency_scan" -eq 1 ] && [ "$total_files" -gt 0 ]; then
build_dependency_graph
# Use optimized compilation order
cp "$temp_dir/compile_order.txt" "$typst_files_list"
log_info "Optimized compilation order based on dependencies"
fi
# 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
# Cache usage statistics (if enabled and not in quiet mode)
if [ "$use_cache" -eq 1 ] && [ "$quiet" -eq 0 ]; then
cache_files=$(find "${cache_dir}/pdfs" -type f | wc -l)
cache_size=$(du -sh "${cache_dir}" 2>/dev/null | cut -f1)
echo -e "\n${CYAN}${ICON_CACHE} Cache statistics:${NC}"
echo -e " - Cached files: ${BOLD}$cache_files${NC}"
echo -e " - Cache size: ${BOLD}$cache_size${NC}"
echo -e " - Cache location: ${BOLD}$cache_dir${NC}"
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 | use_cache=1 # Enable caching by default |
15 | use_git=1 # Enable Git integration by default |
16 | cache_dir="${XDG_CACHE_HOME:-$HOME/.cache}/typst_compiler" |
17 | dependency_scan=1 # Enable dependency scanning by default |
18 | compile_log="typst_compile_errors.log" |
19 | move_log="typst_move_errors.log" |
20 | declare -a exclude_patterns |
21 | |
22 | # Show usage/help information |
23 | show_help() { |
24 | cat << EOF |
25 | Usage: $(basename "$0") [OPTIONS] |
26 | |
27 | Compile Typst files and move generated PDFs to a designated directory. |
28 | |
29 | Options: |
30 | -j, --jobs NUM Number of parallel jobs (default: auto-detect) |
31 | -o, --output-dir DIR Output directory name (default: pdfs) |
32 | -v, --verbose Increase verbosity |
33 | -q, --quiet Suppress most output |
34 | -c, --create-dirs Create output directories if they don't exist |
35 | -d, --dry-run Show what would be done without doing it |
36 | -s, --skip-newer Skip compilation if PDF is newer than source |
37 | -S, --select Interactive selection mode using skim |
38 | -f, --force Force compilation even if PDF exists |
39 | --no-progress Disable progress bar |
40 | --no-cache Disable compilation caching |
41 | --no-git Disable Git-based change detection |
42 | --clear-cache Clear the cache before compiling |
43 | --no-deps Disable dependency scanning |
44 | --cache-dir DIR Custom cache directory (default: ~/.cache/typst_compiler) |
45 | --compile-log FILE Custom location for compilation log (default: $compile_log) |
46 | --move-log FILE Custom location for move log (default: $move_log) |
47 | -e, --exclude PATTERN Exclude files matching pattern (can be used multiple times) |
48 | -h, --help Show this help message and exit |
49 | |
50 | Examples: |
51 | $(basename "$0") -j 4 -o output -c |
52 | $(basename "$0") --verbose --skip-newer |
53 | $(basename "$0") -S # Select files to compile interactively |
54 | $(basename "$0") -e "**/test/**" -e "**/draft/**" |
55 | $(basename "$0") --no-git # Disable Git-based change detection |
56 | EOF |
57 | } |
58 | |
59 | # Parse command line arguments |
60 | while [[ $# -gt 0 ]]; do |
61 | case $1 in |
62 | -j|--jobs) |
63 | jobs="$2" |
64 | shift 2 |
65 | ;; |
66 | -o|--output-dir) |
67 | output_dir="$2" |
68 | shift 2 |
69 | ;; |
70 | -v|--verbose) |
71 | verbose=1 |
72 | shift |
73 | ;; |
74 | -q|--quiet) |
75 | quiet=1 |
76 | shift |
77 | ;; |
78 | -c|--create-dirs) |
79 | create_dirs=1 |
80 | shift |
81 | ;; |
82 | -d|--dry-run) |
83 | dry_run=1 |
84 | shift |
85 | ;; |
86 | -s|--skip-newer) |
87 | skip_newer=1 |
88 | shift |
89 | ;; |
90 | -S|--select) |
91 | select_mode=1 |
92 | shift |
93 | ;; |
94 | -f|--force) |
95 | force=1 |
96 | shift |
97 | ;; |
98 | --no-progress) |
99 | show_progress=0 |
100 | shift |
101 | ;; |
102 | --no-cache) |
103 | use_cache=0 |
104 | shift |
105 | ;; |
106 | --no-git) |
107 | use_git=0 |
108 | shift |
109 | ;; |
110 | --clear-cache) |
111 | rm -rf "${cache_dir}" |
112 | shift |
113 | ;; |
114 | --no-deps) |
115 | dependency_scan=0 |
116 | shift |
117 | ;; |
118 | --cache-dir) |
119 | cache_dir="$2" |
120 | shift 2 |
121 | ;; |
122 | --compile-log) |
123 | compile_log="$2" |
124 | shift 2 |
125 | ;; |
126 | --move-log) |
127 | move_log="$2" |
128 | shift 2 |
129 | ;; |
130 | -e|--exclude) |
131 | exclude_patterns+=("$2") |
132 | shift 2 |
133 | ;; |
134 | -h|--help) |
135 | show_help |
136 | exit 0 |
137 | ;; |
138 | *) |
139 | echo "Unknown option: $1" |
140 | show_help |
141 | exit 1 |
142 | ;; |
143 | esac |
144 | done |
145 | |
146 | # Check for conflicting options |
147 | if [ "$verbose" -eq 1 ] && [ "$quiet" -eq 1 ]; then |
148 | echo "Error: Cannot use both --verbose and --quiet" |
149 | exit 1 |
150 | fi |
151 | |
152 | # Check if required tools are installed |
153 | check_tool() { |
154 | if ! command -v "$1" &> /dev/null; then |
155 | echo "$1 is not installed. Please install it first." |
156 | echo "On most systems: $2" |
157 | exit 1 |
158 | fi |
159 | } |
160 | |
161 | check_tool "fd" "cargo install fd-find or apt/brew install fd-find" |
162 | check_tool "rg" "cargo install ripgrep or apt/brew install ripgrep" |
163 | check_tool "parallel" "apt/brew install parallel" |
164 | check_tool "sha256sum" "Built-in on most Linux systems, on macOS: brew install coreutils" |
165 | |
166 | # Check for skim and bat if select mode is enabled |
167 | if [ "$select_mode" -eq 1 ]; then |
168 | check_tool "sk" "cargo install skim or apt/brew install skim" |
169 | check_tool "bat" "cargo install bat or apt/brew install bat" |
170 | fi |
171 | |
172 | # ANSI color codes |
173 | GREEN='\033[0;32m' |
174 | BLUE='\033[0;34m' |
175 | YELLOW='\033[0;33m' |
176 | RED='\033[0;31m' |
177 | CYAN='\033[0;36m' |
178 | PURPLE='\033[0;35m' |
179 | NC='\033[0m' # No Color |
180 | BOLD='\033[1m' |
181 | |
182 | # Nerd font icons |
183 | ICON_SUCCESS=" " |
184 | ICON_ERROR=" " |
185 | ICON_WORKING=" " |
186 | ICON_COMPILE=" " |
187 | ICON_MOVE=" " |
188 | ICON_COMPLETE=" " |
189 | ICON_SUMMARY=" " |
190 | ICON_INFO=" " |
191 | ICON_DEBUG=" " |
192 | ICON_SELECT=" " |
193 | ICON_CACHE=" " |
194 | ICON_GIT=" " |
195 | |
196 | # Logging functions |
197 | log_debug() { |
198 | if [ "$verbose" -eq 1 ]; then |
199 | echo -e "${BLUE}$ICON_DEBUG${NC} $*" |
200 | fi |
201 | } |
202 | |
203 | log_info() { |
204 | if [ "$quiet" -eq 0 ]; then |
205 | echo -e "${CYAN}$ICON_INFO${NC} $*" |
206 | fi |
207 | } |
208 | |
209 | log_warning() { |
210 | if [ "$quiet" -eq 0 ]; then |
211 | echo -e "${YELLOW}⚠️ ${NC} $*" |
212 | fi |
213 | } |
214 | |
215 | log_error() { |
216 | echo -e "${RED}${BOLD}$ICON_ERROR${NC} $*" |
217 | } |
218 | |
219 | log_success() { |
220 | if [ "$quiet" -eq 0 ]; then |
221 | echo -e "${GREEN}${BOLD}$ICON_SUCCESS${NC} $*" |
222 | fi |
223 | } |
224 | |
225 | # Create a directory for temporary files |
226 | temp_dir=$(mktemp -d) |
227 | compile_failures_dir="$temp_dir/compile_failures" |
228 | move_failures_dir="$temp_dir/move_failures" |
229 | progress_dir="$temp_dir/progress" |
230 | mkdir -p "$compile_failures_dir" "$move_failures_dir" "$progress_dir" |
231 | |
232 | # Create cache directories if caching is enabled |
233 | if [ "$use_cache" -eq 1 ]; then |
234 | mkdir -p "${cache_dir}/hashes" "${cache_dir}/deps" "${cache_dir}/pdfs" |
235 | log_debug "Using cache directory: $cache_dir" |
236 | fi |
237 | |
238 | # Create a lock file for progress updates and a flag for final progress |
239 | progress_lock="/tmp/typst_progress_lock" |
240 | final_progress_file="$temp_dir/final_progress" |
241 | touch "$final_progress_file" |
242 | |
243 | # Initialize log files |
244 | > "$compile_log" |
245 | > "$move_log" |
246 | |
247 | # Store current working directory |
248 | CWD=$(pwd) |
249 | |
250 | log_info "Starting Typst compilation process..." |
251 | |
252 | # Check if we're in a Git repository |
253 | in_git_repo=0 |
254 | git_repo_root="" |
255 | |
256 | check_git_repo() { |
257 | if [ "$use_git" -eq 1 ] && command -v git &> /dev/null; then |
258 | # Check if we're in a git repo |
259 | if git_repo_root=$(git rev-parse --show-toplevel 2>/dev/null); then |
260 | in_git_repo=1 |
261 | log_info "${ICON_GIT} Git repository detected at: ${BOLD}$git_repo_root${NC}" |
262 | return 0 |
263 | else |
264 | log_debug "Not in a Git repository, falling back to file hashing" |
265 | return 1 |
266 | fi |
267 | else |
268 | log_debug "Git integration disabled or git not found" |
269 | return 1 |
270 | fi |
271 | } |
272 | |
273 | # Get list of changed files from Git |
274 | get_git_changed_files() { |
275 | local git_files="$temp_dir/git_changed_files.txt" |
276 | |
277 | # Get list of staged, unstaged, and untracked files |
278 | git -C "$git_repo_root" ls-files --modified --others --exclude-standard > "$git_files" |
279 | # Add staged files that may not show as modified |
280 | git -C "$git_repo_root" diff --name-only --cached >> "$git_files" |
281 | |
282 | # If no previous commit exists, consider all tracked files as changed |
283 | if ! git -C "$git_repo_root" rev-parse HEAD &>/dev/null; then |
284 | git -C "$git_repo_root" ls-files >> "$git_files" |
285 | fi |
286 | |
287 | # Filter to keep only .typ files and remove duplicates |
288 | sort -u "$git_files" | grep '\.typ$' > "$temp_dir/git_changed_typ_files.txt" |
289 | |
290 | log_debug "Git reports $(wc -l < "$temp_dir/git_changed_typ_files.txt") changed .typ files" |
291 | } |
292 | |
293 | # Build exclude arguments for fd |
294 | fd_exclude_args=() |
295 | for pattern in "${exclude_patterns[@]}"; do |
296 | fd_exclude_args+=("-E" "$pattern") |
297 | done |
298 | |
299 | # Create a list of files to process |
300 | typst_files_list="$temp_dir/typst_files.txt" |
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="$temp_dir/bat_config" |
308 | mkdir -p "$bat_config/syntaxes" |
309 | export BAT_CONFIG_PATH="$bat_config" |
310 | |
311 | # Use skim with bat for preview to select files |
312 | selected_files="$temp_dir/selected_files.txt" |
313 | |
314 | # Find all eligible .typ files for selection |
315 | fd '\.typ$' --type f "${fd_exclude_args[@]}" . > "$typst_files_list.all" |
316 | |
317 | # Check if we found any files |
318 | if [ ! -s "$typst_files_list.all" ]; then |
319 | log_error "No .typ files found matching your criteria." |
320 | rm -rf "$temp_dir" |
321 | exit 0 |
322 | fi |
323 | |
324 | # Prepare preview command for skim: use bat with custom syntax for .typ files |
325 | preview_cmd="bat --color=always --style=numbers --map-syntax='*.typ:Markdown' {} 2>/dev/null || cat {}" |
326 | |
327 | # Use skim with bat preview to select files |
328 | cat "$typst_files_list.all" | sk --multi \ |
329 | --preview "$preview_cmd" \ |
330 | --preview-window "right:70%" \ |
331 | --height "80%" \ |
332 | --prompt "Select .typ files to compile: " \ |
333 | --header "TAB: Select multiple files, ENTER: Confirm, CTRL-C: Cancel" \ |
334 | --no-mouse > "$selected_files" |
335 | |
336 | # Check if user selected any files |
337 | if [ ! -s "$selected_files" ]; then |
338 | log_error "No files selected. Exiting." |
339 | rm -rf "$temp_dir" |
340 | exit 0 |
341 | fi |
342 | |
343 | # Use the selected files instead of all discovered files |
344 | cp "$selected_files" "$typst_files_list" |
345 | total_selected=$(wc -l < "$typst_files_list") |
346 | total_available=$(wc -l < "$typst_files_list.all") |
347 | |
348 | log_info "Selected ${BOLD}$total_selected${NC} out of ${BOLD}$total_available${NC} available files" |
349 | else |
350 | # Normal mode - process all files |
351 | fd '\.typ$' --type f "${fd_exclude_args[@]}" . > "$typst_files_list" |
352 | log_info "Found ${BOLD}$(wc -l < "$typst_files_list")${NC} Typst files to process" |
353 | |
354 | # Check for Git repository and get changed files |
355 | check_git_repo |
356 | if [ "$in_git_repo" -eq 1 ] && [ "$force" -eq 0 ]; then |
357 | get_git_changed_files |
358 | |
359 | # If no forced compilation, filter the file list to only include changed files |
360 | if [ -s "$temp_dir/git_changed_typ_files.txt" ]; then |
361 | cp "$typst_files_list" "$temp_dir/all_typ_files.txt" |
362 | |
363 | # Convert Git changed files to absolute paths |
364 | while read -r file; do |
365 | echo "$git_repo_root/$file" |
366 | done < "$temp_dir/git_changed_typ_files.txt" | sort > "$temp_dir/git_absolute_paths.txt" |
367 | |
368 | # Find the intersection of all files and Git changed files |
369 | comm -12 <(sort "$typst_files_list") "$temp_dir/git_absolute_paths.txt" > "$temp_dir/files_to_compile.txt" |
370 | |
371 | # Use the filtered list if there are changes, otherwise keep using all files |
372 | if [ -s "$temp_dir/files_to_compile.txt" ]; then |
373 | cp "$temp_dir/files_to_compile.txt" "$typst_files_list" |
374 | git_changes_count=$(wc -l < "$typst_files_list") |
375 | log_info "${ICON_GIT} Processing ${BOLD}$git_changes_count${NC} files with Git changes" |
376 | else |
377 | log_info "${ICON_GIT} No Git changes detected for .typ files" |
378 | |
379 | # If no changes detected and not forced, update progress and exit early |
380 | if [ "$force" -eq 0 ]; then |
381 | log_success "No files need to be compiled." |
382 | rm -rf "$temp_dir" |
383 | exit 0 |
384 | fi |
385 | fi |
386 | fi |
387 | fi |
388 | fi |
389 | |
390 | total_files=$(wc -l < "$typst_files_list") |
391 | |
392 | # Function to extract Typst imports from a file |
393 | extract_dependencies() { |
394 | local file="$1" |
395 | # Match both import formats: #import "file.typ" and #include "file.typ" |
396 | # Also handle package imports: #import "@preview:0.1.0" |
397 | # Return full paths to local imports, skipping package imports |
398 | rg -U "#(import|include) \"(?!@)([^\"]+)\"" -r "$CWD/\$2" "$file" | \ |
399 | while read -r dep; do |
400 | # Resolve relative paths correctly |
401 | if [[ "$dep" != /* ]]; then |
402 | # If not absolute path, make it relative to the file's directory |
403 | file_dir=$(dirname "$file") |
404 | dep="$file_dir/$dep" |
405 | fi |
406 | |
407 | # Normalize path |
408 | dep=$(realpath --relative-to="$CWD" "$dep" 2>/dev/null || echo "$dep") |
409 | |
410 | # Only output if the dependency exists and is a .typ file |
411 | if [[ "$dep" == *.typ ]] && [ -f "$dep" ]; then |
412 | echo "$dep" |
413 | fi |
414 | done | sort -u # Sort and remove duplicates |
415 | } |
416 | |
417 | # Function to build dependency graph for all files |
418 | build_dependency_graph() { |
419 | log_info "${ICON_CACHE} Building dependency graph..." |
420 | |
421 | # Create a deps file for each Typst file |
422 | while read -r file; do |
423 | file_hash=$(echo "$file" | md5sum | cut -d' ' -f1) |
424 | deps_file="${cache_dir}/deps/${file_hash}.deps" |
425 | |
426 | # Skip if deps file is fresh and not forced |
427 | if [ "$force" -eq 0 ] && [ -f "$deps_file" ] && [ "$file" -ot "$deps_file" ]; then |
428 | log_debug "Using cached dependencies for $file" |
429 | continue |
430 | fi |
431 | |
432 | # Extract dependencies and save to deps file |
433 | extract_dependencies "$file" > "$deps_file" |
434 | log_debug "Updated dependencies for $file" |
435 | done < "$typst_files_list" |
436 | |
437 | # Create sorted compilation order based on dependencies |
438 | # Files with fewer dependencies (or no dependencies) come first |
439 | if [ -f "$temp_dir/compile_order.txt" ]; then |
440 | rm "$temp_dir/compile_order.txt" |
441 | fi |
442 | |
443 | # First pass: count dependencies for each file |
444 | declare -A dep_counts |
445 | while read -r file; do |
446 | file_hash=$(echo "$file" | md5sum | cut -d' ' -f1) |
447 | deps_file="${cache_dir}/deps/${file_hash}.deps" |
448 | |
449 | # Count dependencies |
450 | if [ -f "$deps_file" ]; then |
451 | dep_count=$(wc -l < "$deps_file") |
452 | else |
453 | dep_count=0 |
454 | fi |
455 | |
456 | dep_counts["$file"]=$dep_count |
457 | done < "$typst_files_list" |
458 | |
459 | # Second pass: sort files by dependency count (ascending) |
460 | while read -r file; do |
461 | echo "${dep_counts["$file"]} $file" |
462 | done < "$typst_files_list" | sort -n | cut -d' ' -f2- > "$temp_dir/compile_order.txt" |
463 | |
464 | log_debug "Created optimized compilation order" |
465 | |
466 | # If in a Git repo, consider dependent files of changed files |
467 | if [ "$in_git_repo" -eq 1 ] && [ -f "$temp_dir/git_changed_typ_files.txt" ]; then |
468 | log_info "${ICON_GIT} Analyzing dependencies of changed files..." |
469 | |
470 | # Track files to be compiled (start with directly changed files) |
471 | cp "$typst_files_list" "$temp_dir/files_to_compile_with_deps.txt" |
472 | |
473 | # Identify files that depend on changed files (reverse dependencies) |
474 | while read -r file; do |
475 | file_hash=$(echo "$file" | md5sum | cut -d' ' -f1) |
476 | |
477 | # Find all files that have this file as a dependency |
478 | grep -l "^$file$" "${cache_dir}"/deps/*.deps 2>/dev/null | while read -r dep_file; do |
479 | # Extract the dependent file from the hash |
480 | dependent_hash=$(basename "$dep_file" .deps) |
481 | |
482 | # Find the original filename for this hash |
483 | while read -r potential_file; do |
484 | potential_hash=$(echo "$potential_file" | md5sum | cut -d' ' -f1) |
485 | if [ "$potential_hash" == "$dependent_hash" ]; then |
486 | # Add this dependent file to the compilation list |
487 | echo "$potential_file" >> "$temp_dir/files_to_compile_with_deps.txt" |
488 | break |
489 | fi |
490 | done < "$temp_dir/all_typ_files.txt" |
491 | done |
492 | done < "$typst_files_list" |
493 | |
494 | # Remove duplicates and update the files list |
495 | sort -u "$temp_dir/files_to_compile_with_deps.txt" > "$typst_files_list" |
496 | |
497 | # Update the total count |
498 | new_total=$(wc -l < "$typst_files_list") |
499 | if [ "$new_total" -gt "$total_files" ]; then |
500 | log_info "${ICON_GIT} Added ${BOLD}$((new_total - total_files))${NC} additional files with dependencies" |
501 | total_files=$new_total |
502 | fi |
503 | fi |
504 | } |
505 | |
506 | # Function to check if a file has changed since last compilation |
507 | has_file_changed() { |
508 | local file="$1" |
509 | |
510 | # Force recompilation if requested |
511 | if [ "$force" -eq 1 ]; then |
512 | log_debug "Force recompilation of $file" |
513 | return 0 # File considered changed |
514 | fi |
515 | |
516 | # If using Git and in a Git repo, check if file is in the changed list |
517 | if [ "$in_git_repo" -eq 1 ] && [ "$use_git" -eq 1 ]; then |
518 | # Convert to repo-relative path |
519 | local repo_path="${file#$git_repo_root/}" |
520 | |
521 | # Check if this file or any of its dependencies are changed according to Git |
522 | if grep -q "^$git_repo_root/$repo_path$" "$temp_dir/git_absolute_paths.txt" 2>/dev/null; then |
523 | log_debug "Git reports $file has changed" |
524 | return 0 # File considered changed |
525 | fi |
526 | |
527 | # If not in the changed list and we're only processing Git changes, |
528 | # then this file must have a dependency that changed |
529 | if [ -f "$temp_dir/git_changed_typ_files.txt" ]; then |
530 | # Check dependencies (if enabled) |
531 | if [ "$dependency_scan" -eq 1 ]; then |
532 | local file_hash=$(echo "$file" | md5sum | cut -d' ' -f1) |
533 | local deps_file="${cache_dir}/deps/${file_hash}.deps" |
534 | |
535 | if [ -f "$deps_file" ]; then |
536 | while read -r dep; do |
537 | if has_file_changed "$dep"; then |
538 | log_debug "Dependency $dep of $file has changed" |
539 | return 0 # Dependency has changed |
540 | fi |
541 | done < "$deps_file" |
542 | fi |
543 | fi |
544 | |
545 | log_debug "No changes to $file or its dependencies according to Git" |
546 | return 1 # No changes detected via Git |
547 | fi |
548 | fi |
549 | |
550 | # Fall back to hash-based detection if Git not available or not conclusive |
551 | local file_hash=$(echo "$file" | md5sum | cut -d' ' -f1) |
552 | local hash_file="${cache_dir}/hashes/${file_hash}.sha256" |
553 | |
554 | # Check if hash file exists |
555 | if [ ! -f "$hash_file" ]; then |
556 | log_debug "No previous hash for $file" |
557 | return 0 # File considered changed |
558 | fi |
559 | |
560 | # Compare current file hash with stored hash |
561 | local current_hash=$(sha256sum "$file" | cut -d' ' -f1) |
562 | local stored_hash=$(cat "$hash_file") |
563 | |
564 | if [ "$current_hash" != "$stored_hash" ]; then |
565 | log_debug "File $file has changed since last compilation" |
566 | return 0 # File has changed |
567 | fi |
568 | |
569 | # Check if any dependencies have changed |
570 | if [ "$dependency_scan" -eq 1 ]; then |
571 | local deps_file="${cache_dir}/deps/${file_hash}.deps" |
572 | if [ -f "$deps_file" ]; then |
573 | while read -r dep; do |
574 | if has_file_changed "$dep"; then |
575 | log_debug "Dependency $dep of $file has changed" |
576 | return 0 # Dependency has changed |
577 | fi |
578 | done < "$deps_file" |
579 | fi |
580 | fi |
581 | |
582 | log_debug "File $file and its dependencies are unchanged" |
583 | return 1 # File and dependencies haven't changed |
584 | } |
585 | |
586 | # Function to update file hash after compilation |
587 | update_file_hash() { |
588 | local file="$1" |
589 | local file_hash=$(echo "$file" | md5sum | cut -d' ' -f1) |
590 | local hash_file="${cache_dir}/hashes/${file_hash}.sha256" |
591 | |
592 | # Update hash file |
593 | sha256sum "$file" | cut -d' ' -f1 > "$hash_file" |
594 | log_debug "Updated hash for $file" |
595 | } |
596 | |
597 | # Function to cache compiled PDF |
598 | cache_pdf() { |
599 | local src_file="$1" |
600 | local pdf_file="$2" |
601 | |
602 | if [ ! -f "$pdf_file" ]; then |
603 | log_debug "No PDF file to cache: $pdf_file" |
604 | return 1 |
605 | fi |
606 | |
607 | local file_hash=$(echo "$src_file" | md5sum | cut -d' ' -f1) |
608 | local cached_pdf="${cache_dir}/pdfs/${file_hash}.pdf" |
609 | |
610 | # Copy PDF to cache |
611 | cp "$pdf_file" "$cached_pdf" |
612 | log_debug "Cached PDF for $src_file" |
613 | return 0 |
614 | } |
615 | |
616 | # Function to retrieve cached PDF |
617 | get_cached_pdf() { |
618 | local src_file="$1" |
619 | local target_file="$2" |
620 | |
621 | local file_hash=$(echo "$src_file" | md5sum | cut -d' ' -f1) |
622 | local cached_pdf="${cache_dir}/pdfs/${file_hash}.pdf" |
623 | |
624 | if [ ! -f "$cached_pdf" ]; then |
625 | log_debug "No cached PDF for $src_file" |
626 | return 1 |
627 | fi |
628 | |
629 | # Copy cached PDF to target location |
630 | cp "$cached_pdf" "$target_file" |
631 | log_debug "Retrieved cached PDF for $src_file" |
632 | return 0 |
633 | } |
634 | |
635 | # Function to update progress bar during processing |
636 | update_progress_during() { |
637 | # If progress is disabled, do nothing |
638 | if [ "$show_progress" -eq 0 ]; then |
639 | return 0 |
640 | fi |
641 | |
642 | ( |
643 | # Try to acquire lock, but don't wait if busy |
644 | flock -n 200 || return 0 |
645 | |
646 | completed=$(find "$progress_dir" -type f | wc -l) |
647 | percent=$((completed * 100 / total_files)) |
648 | bar_length=50 |
649 | filled_length=$((bar_length * completed / total_files)) |
650 | |
651 | # Create the progress bar |
652 | bar="" |
653 | for ((i=0; i<bar_length; i++)); do |
654 | if [ $i -lt $filled_length ]; then |
655 | bar="${bar}█" |
656 | else |
657 | bar="${bar}░" |
658 | fi |
659 | done |
660 | |
661 | # Calculate success and failure counts |
662 | success=$((completed - $(find "$compile_failures_dir" "$move_failures_dir" -type f | wc -l))) |
663 | compile_fails=$(find "$compile_failures_dir" -type f | wc -l) |
664 | move_fails=$(find "$move_failures_dir" -type f | wc -l) |
665 | |
666 | # Clear the previous line and print the updated progress |
667 | echo -ne "\r\033[K" |
668 | echo -ne "${PURPLE}$ICON_WORKING Progress: ${GREEN}$bar ${BOLD}${percent}%${NC} " |
669 | echo -ne "[${GREEN}${success}✓${NC}|${RED}${compile_fails}✗${NC}|${YELLOW}${move_fails}!${NC}] " |
670 | echo -ne "${BLUE}($completed/$total_files)${NC}" |
671 | |
672 | # If all files are processed, print the completion message ONLY ONCE |
673 | if [ $completed -eq $total_files ] && [ ! -e "$final_progress_file.done" ]; then |
674 | touch "$final_progress_file.done" |
675 | echo -e "\n${GREEN}${BOLD}$ICON_COMPLETE All files processed!${NC}" |
676 | fi |
677 | ) 200>"$progress_lock" |
678 | } |
679 | |
680 | # Function to show final progress (called only once at the end) |
681 | show_final_progress() { |
682 | # If progress is disabled, do nothing |
683 | if [ "$show_progress" -eq 0 ]; then |
684 | return 0 |
685 | fi |
686 | |
687 | ( |
688 | flock -w 1 200 |
689 | |
690 | # Only proceed if the final progress hasn't been shown yet |
691 | if [ -e "$final_progress_file.done" ]; then |
692 | return 0 |
693 | fi |
694 | |
695 | completed=$(find "$progress_dir" -type f | wc -l) |
696 | percent=$((completed * 100 / total_files)) |
697 | bar_length=50 |
698 | filled_length=$((bar_length * completed / total_files)) |
699 | |
700 | # Create the progress bar |
701 | bar="" |
702 | for ((i=0; i<bar_length; i++)); do |
703 | if [ $i -lt $filled_length ]; then |
704 | bar="${bar}█" |
705 | else |
706 | bar="${bar}░" |
707 | fi |
708 | done |
709 | |
710 | # Calculate success and failure counts |
711 | success=$((completed - $(find "$compile_failures_dir" "$move_failures_dir" -type f | wc -l))) |
712 | compile_fails=$(find "$compile_failures_dir" -type f | wc -l) |
713 | move_fails=$(find "$move_failures_dir" -type f | wc -l) |
714 | |
715 | # Clear the previous line and print the updated progress |
716 | echo -ne "\r\033[K" |
717 | echo -ne "${PURPLE}$ICON_WORKING Progress: ${GREEN}$bar ${BOLD}${percent}%${NC} " |
718 | echo -ne "[${GREEN}${success}✓${NC}|${RED}${compile_fails}✗${NC}|${YELLOW}${move_fails}!${NC}] " |
719 | echo -ne "${BLUE}($completed/$total_files)${NC}" |
720 | |
721 | # Mark final progress as shown |
722 | touch "$final_progress_file.done" |
723 | |
724 | # If all files are processed, print the completion message |
725 | if [ $completed -eq $total_files ]; then |
726 | echo -e "\n${GREEN}${BOLD}$ICON_COMPLETE All files processed!${NC}" |
727 | fi |
728 | ) 200>"$progress_lock" |
729 | } |
730 | |
731 | # Function to process a single .typ file |
732 | process_file() { |
733 | typfile="$1" |
734 | file_id=$(echo "$typfile" | md5sum | cut -d' ' -f1) |
735 | |
736 | # Get the directory containing the .typ file |
737 | typdir=$(dirname "$typfile") |
738 | # Get the filename without path |
739 | filename=$(basename "$typfile") |
740 | # Get the filename without extension |
741 | basename="${filename%.typ}" |
742 | |
743 | # Check if output directory exists or should be created |
744 | target_dir="$typdir/$output_dir" |
745 | if [ ! -d "$target_dir" ]; then |
746 | if [ "$create_dirs" -eq 1 ]; then |
747 | if [ "$dry_run" -eq 0 ]; then |
748 | mkdir -p "$target_dir" |
749 | log_debug "Created directory: $target_dir" |
750 | else |
751 | log_debug "[DRY RUN] Would create directory: $target_dir" |
752 | fi |
753 | else |
754 | # Skip this file if output directory doesn't exist and --create-dirs not specified |
755 | log_debug "Skipping $typfile (no $output_dir directory)" |
756 | touch "$progress_dir/$file_id" |
757 | update_progress_during |
758 | return 0 |
759 | fi |
760 | fi |
761 | |
762 | target_pdf="$target_dir/$basename.pdf" |
763 | |
764 | # Skip if PDF is newer than source and --skip-newer is specified |
765 | if [ "$skip_newer" -eq 1 ] && [ -f "$target_pdf" ]; then |
766 | if [ "$typfile" -ot "$target_pdf" ] && [ "$force" -eq 0 ]; then |
767 | log_debug "Skipping $typfile (PDF is newer)" |
768 | touch "$progress_dir/$file_id" |
769 | update_progress_during |
770 | return 0 |
771 | fi |
772 | fi |
773 | |
774 | # Check if file has changed (if caching is enabled) |
775 | if [ "$use_cache" -eq 1 ] && [ "$dry_run" -eq 0 ]; then |
776 | if ! has_file_changed "$typfile"; then |
777 | # Try to retrieve cached PDF |
778 | if get_cached_pdf "$typfile" "$target_pdf"; then |
779 | log_debug "Using cached PDF for $typfile" |
780 | touch "$progress_dir/$file_id" |
781 | update_progress_during |
782 | return 0 |
783 | fi |
784 | fi |
785 | fi |
786 | |
787 | # Create a temporary file for capturing compiler output |
788 | temp_output="$temp_dir/output_${file_id}.log" |
789 | |
790 | # Add a header to the log before compilation |
791 | { |
792 | echo -e "\n===== COMPILING: $typfile =====" |
793 | echo "$(date)" |
794 | } > "$temp_output.header" |
795 | |
796 | # In dry run mode, just log what would be done |
797 | if [ "$dry_run" -eq 1 ]; then |
798 | log_debug "[DRY RUN] Would compile: $typfile" |
799 | log_debug "[DRY RUN] Would move to: $target_pdf" |
800 | touch "$progress_dir/$file_id" |
801 | update_progress_during |
802 | return 0 |
803 | fi |
804 | |
805 | # Compile the .typ file using typst with --root flag and capture all output |
806 | if ! typst compile --root "$CWD" "$typfile" > "$temp_output.stdout" 2> "$temp_output.stderr"; then |
807 | # Store the failure |
808 | echo "$typfile" > "$compile_failures_dir/$file_id" |
809 | log_debug "Compilation failed for $typfile" |
810 | |
811 | # Combine stdout and stderr |
812 | cat "$temp_output.stdout" "$temp_output.stderr" > "$temp_output.combined" |
813 | |
814 | # Filter the output to only include error messages using ripgrep |
815 | rg "error:" -A 20 "$temp_output.combined" > "$temp_output.errors" || true |
816 | |
817 | # Lock the log file to avoid concurrent writes corrupting it |
818 | ( |
819 | flock -w 1 201 |
820 | cat "$temp_output.header" "$temp_output.errors" >> "$compile_log" |
821 | echo -e "\n" >> "$compile_log" |
822 | ) 201>"$compile_log.lock" |
823 | else |
824 | # Check if the output PDF exists |
825 | temp_pdf="$typdir/$basename.pdf" |
826 | if [ -f "$temp_pdf" ]; then |
827 | # Cache the PDF if caching is enabled |
828 | if [ "$use_cache" -eq 1 ]; then |
829 | cache_pdf "$typfile" "$temp_pdf" |
830 | # Update file hash |
831 | update_file_hash "$typfile" |
832 | fi |
833 | |
834 | # Try to move the output PDF to the output directory |
835 | move_header="$temp_dir/move_${file_id}.header" |
836 | { |
837 | echo -e "\n===== MOVING: $typfile =====" |
838 | echo "$(date)" |
839 | } > "$move_header" |
840 | |
841 | if ! mv "$temp_pdf" "$target_dir/" 2> "$temp_output.move_err"; then |
842 | echo "$typfile -> $target_pdf" > "$move_failures_dir/$file_id" |
843 | log_debug "Failed to move $basename.pdf to $target_dir/" |
844 | |
845 | # Lock the log file to avoid concurrent writes corrupting it |
846 | ( |
847 | flock -w 1 202 |
848 | cat "$move_header" "$temp_output.move_err" >> "$move_log" |
849 | echo "Failed to move $temp_pdf to $target_dir/" >> "$move_log" |
850 | ) 202>"$move_log.lock" |
851 | else |
852 | log_debug "Moved $basename.pdf to $target_dir/" |
853 | fi |
854 | else |
855 | # This is a fallback check in case typst doesn't return error code |
856 | echo "$typfile" > "$compile_failures_dir/$file_id" |
857 | log_debug "Compilation completed without errors but no PDF was generated for $typfile" |
858 | |
859 | # Lock the log file to avoid concurrent writes corrupting it |
860 | ( |
861 | flock -w 1 201 |
862 | echo "Compilation completed without errors but no PDF was generated" >> "$compile_log" |
863 | ) 201>"$compile_log.lock" |
864 | fi |
865 | fi |
866 | |
867 | # Mark this file as processed (for progress tracking) |
868 | touch "$progress_dir/$file_id" |
869 | |
870 | # Update the progress bar |
871 | update_progress_during |
872 | } |
873 | |
874 | export -f process_file |
875 | export -f update_progress_during |
876 | export -f show_final_progress |
877 | export -f log_debug |
878 | export -f log_info |
879 | export -f log_warning |
880 | export -f log_error |
881 | export -f log_success |
882 | export -f has_file_changed |
883 | export -f update_file_hash |
884 | export -f cache_pdf |
885 | export -f get_cached_pdf |
886 | export CWD |
887 | export temp_dir |
888 | export compile_failures_dir |
889 | export move_failures_dir |
890 | export progress_dir |
891 | export final_progress_file |
892 | export progress_lock |
893 | export compile_log |
894 | export move_log |
895 | export cache_dir |
896 | export git_repo_root |
897 | export in_git_repo |
898 | export total_files |
899 | export GREEN BLUE YELLOW RED CYAN PURPLE NC BOLD |
900 | export ICON_SUCCESS ICON_ERROR ICON_WORKING ICON_COMPILE ICON_MOVE ICON_COMPLETE ICON_SUMMARY ICON_INFO ICON_DEBUG ICON_CACHE ICON_GIT |
901 | export verbose |
902 | export quiet |
903 | export output_dir |
904 | export create_dirs |
905 | export dry_run |
906 | export skip_newer |
907 | export show_progress |
908 | export force |
909 | export use_cache |
910 | export dependency_scan |
911 | export use_git |
912 | |
913 | # Determine the number of CPU cores and use that many parallel jobs (if not specified) |
914 | if [ "$jobs" -eq 0 ]; then |
915 | jobs=$(nproc 2>/dev/null || sysctl -n hw.ncpu 2>/dev/null || echo 4) |
916 | fi |
917 | log_info "Using ${BOLD}$jobs${NC} parallel jobs for compilation" |
918 | |
919 | # Build dependency graph if dependency scanning is enabled |
920 | if [ "$dependency_scan" -eq 1 ] && [ "$total_files" -gt 0 ]; then |
921 | build_dependency_graph |
922 | # Use optimized compilation order |
923 | cp "$temp_dir/compile_order.txt" "$typst_files_list" |
924 | log_info "Optimized compilation order based on dependencies" |
925 | fi |
926 | |
927 | # Initialize progress bar if showing progress |
928 | if [ "$show_progress" -eq 1 ]; then |
929 | update_progress_during |
930 | fi |
931 | |
932 | # Process files in parallel with --will-cite to suppress citation notice |
933 | if [ "$total_files" -gt 0 ]; then |
934 | cat "$typst_files_list" | parallel --will-cite --jobs "$jobs" process_file |
935 | |
936 | # Wait a moment for any remaining progress updates to complete |
937 | sleep 0.5 |
938 | |
939 | # Show the final progress exactly once if showing progress |
940 | if [ "$show_progress" -eq 1 ]; then |
941 | show_final_progress |
942 | fi |
943 | |
944 | # Print summary of failures |
945 | if [ "$quiet" -eq 0 ]; then |
946 | echo -e "\n${BOLD}${PURPLE}$ICON_SUMMARY Processing Summary${NC}" |
947 | fi |
948 | |
949 | # Collect all failure files |
950 | compile_failures=$(find "$compile_failures_dir" -type f | wc -l) |
951 | move_failures=$(find "$move_failures_dir" -type f | wc -l) |
952 | |
953 | if [ "$compile_failures" -eq 0 ] && [ "$move_failures" -eq 0 ]; then |
954 | log_success "All files processed successfully." |
955 | else |
956 | if [ "$compile_failures" -gt 0 ]; then |
957 | echo -e "\n${RED}${BOLD}$ICON_ERROR Compilation failures:${NC}" |
958 | find "$compile_failures_dir" -type f -exec cat {} \; | sort | while read -r failure; do |
959 | echo -e "${RED}- $failure${NC}" |
960 | done |
961 | echo -e "${BLUE}See $compile_log for detailed error messages.${NC}" |
962 | fi |
963 | |
964 | if [ "$move_failures" -gt 0 ]; then |
965 | echo -e "\n${YELLOW}${BOLD}$ICON_ERROR Move failures:${NC}" |
966 | find "$move_failures_dir" -type f -exec cat {} \; | sort | while read -r failure; do |
967 | echo -e "${YELLOW}- $failure${NC}" |
968 | done |
969 | echo -e "${BLUE}See $move_log for detailed error messages.${NC}" |
970 | fi |
971 | fi |
972 | |
973 | # Cache usage statistics (if enabled and not in quiet mode) |
974 | if [ "$use_cache" -eq 1 ] && [ "$quiet" -eq 0 ]; then |
975 | cache_files=$(find "${cache_dir}/pdfs" -type f | wc -l) |
976 | cache_size=$(du -sh "${cache_dir}" 2>/dev/null | cut -f1) |
977 | echo -e "\n${CYAN}${ICON_CACHE} Cache statistics:${NC}" |
978 | echo -e " - Cached files: ${BOLD}$cache_files${NC}" |
979 | echo -e " - Cache size: ${BOLD}$cache_size${NC}" |
980 | echo -e " - Cache location: ${BOLD}$cache_dir${NC}" |
981 | fi |
982 | else |
983 | log_warning "No .typ files found to process." |
984 | fi |
985 | |
986 | # Clean up temporary directory and lock files |
987 | rm -rf "$temp_dir" |
988 | rm -f "$progress_lock" "$compile_log.lock" "$move_log.lock" |
989 | |
990 | log_success "Processing complete." |
991 |