12P by neo 2일전 | ★ favorite | 댓글 1개
  • BashZsh에서 이미 완성된 단어에도 설명을 표시하는 탭 자동완성 기능을 구현하는 방법 소개
  • Bash와 Zsh는 서로 다른 탭 자동완성 API를 사용하며, Zsh만 기본적으로 자동 완성에 설명 보이기 기능을 제공함
  • _generate_foo_completions로 후보를 생성하고, Bash에서는 COMPREPLY, Zsh에서는 compadd로 반환하는 구조를 구현함
  • Zsh의 설명 기능을 Bash에도 구현하기 위해 후보 문자열에 설명을 포함하고, 단일 후보일 때만 설명을 제거하는 방식으로 처리함
  • 단일 후보일 경우에도 의도적으로 모호성을 추가<TAB> 입력 시 설명이 표시되도록 개선함
  • 최종 스크립트는 두 쉘에서 동일한 사용자 경험을 제공하며, 부분 완성·전체 완성·후보 설명 표시 모두 지원함

문제 배경

  • 탭 자동완성(tab-completion) 은 명령어나 플래그를 탐색할 때 유용하며, 특히 API나 CLI 도구를 처음 접할 때 도움을 줌
  • Zsh는 기본적으로 여러 후보가 있을 때만 설명을 표시하고, Bash는 별도 설정을 통해서만 가능
  • 하지만 이미 완성된 단어에 대해서는 두 셸 모두 설명을 표시하지 않음
  • 이로 인해 사용자가 설명을 보려면 다음과 같은 번거로운 과정을 거쳐야 함
    • 일부 문자를 삭제하여 여러 후보가 매칭되도록 함
    • <TAB> 키를 눌러 후보 목록을 확인
    • 원하는 설명을 시각적으로 찾음
    • 삭제한 문자를 다시 입력하여 명령 실행

해결 방법 개요

  • 단일 후보일 경우에도 더미 후보(dummy completion) 를 추가하여 후보를 모호하게 만듦
  • 이렇게 하면 Bash와 Zsh 모두 후보 목록과 함께 설명을 출력하게 됨
  • 단, 실제 명령어에 설명 텍스트가 삽입되지 않도록 주의해야 함

기본 개념

  • 탭 자동완성은 <TAB> 입력 시 현재 단어와 커서 위치를 받아, 가능한 후보 목록을 반환하는 방식으로 동작함
  • Bash: COMPREPLY 배열에 후보를 할당
  • Zsh: compadd 명령으로 후보를 등록
  • _generate_foo_completions 함수는 후보 문자열을 출력하며, 실전에서는 CLI의 상태를 기반으로 동적 생성 가능

Bash와 Zsh 동시 지원하기

  • _complete_foo_bash_complete_foo_zsh 함수로 각각의 쉘에 맞게 구현
  • if [ -n "${ZSH_VERSION:-}" ]; then ... elif [ -n "${BASH_VERSION:-}" ]; then ... fi로 구분
  • 사용자는 스크립트를 .bashrc.zshrc에 등록 후 적용

Zsh에서의 설명 표시

  • 후보 문자열에 이름: 설명 형식 사용
  • Zsh: compadd -d raw -- $trimmed로 이름과 설명을 병렬 배열로 전달
  • Bash: 설명 부분을 제거한 후보만 COMPREPLY에 전달 (기본적으로 설명 미지원)

Bash에서 설명 구현하기

  • 여러 후보일 경우 설명이 포함된 문자열을 그대로 노출
  • 단일 후보일 경우에만 설명 제거
  • Bash의 자동완성이 공통 접두어만 삽입하는 동작을 이용해 설명이 실제 입력에는 포함되지 않도록 함

단일 후보에서도 설명 표시

  • 완성된 단어에 <TAB> 입력 시에도 설명을 보여주기 위해 더미 후보를 추가해 모호성을 유도
  • Zsh: 두 개의 병렬 배열(raw, trimmed) 모두에 더미 후보 추가
  • Bash: 단일 후보일 경우 trimmed에 이름만 추가

최종 결과

  • 다중 후보 시 이름+설명 모두 표시
  • 단일 후보 시에도 <TAB>로 설명 확인 가능
  • Bash와 Zsh 모두 동일한 경험 제공
  • 적용 예:
    $ foo <TAB>  
    apple: a common fruit banana: starchy and high in potassium  
    apricot: sour fruit... cherry: small and sweet...  
    
Hacker News 의견
  • fish에서 관심 있는 프로그램이 man 페이지를 제공한다면 fish_update_completions만 실행하면 됨
    • 이 명령은 시스템의 모든 man 페이지를 파싱해 자동 완성 파일을 ~/.cache/fish/generated_completions/에 생성함
    • man 페이지가 부실하거나 없으면 직접 작성해 upstream에 기여할 수 있음
    • fish의 포맷은 단순해서 공식 문서만 보면 충분함
    • 예: curl-L → 'Follow redirects', -O → 'Write output to file named as remote file'
    • 화면 공유할 때 사람들이 내가 zsh와 플러그인들을 쓰는 줄 아는데, 사실은 기본 설정만으로도 예쁜 fish를 쓰고 있음
    • 하지만 car TABblkdiscard로 확장되는 비접두사 자동 완성 때문에 cargo가 PATH에 없어도 오작동하는 점은 개선되면 좋겠음
    • man 페이지 없이 --help만 제공하는 프로그램의 경우, fish에 zsh의 _gnu_generic이나 bash의 complete -F _longopt 같은 기능이 있는지 궁금함
    • zsh에서도 man 페이지 기반 자동 완성을 생성하는 스크립트가 있음 → zsh-manpage-completion-generator
    • OpenSUSE에서 zypper search fish-completion을 치면 200개가 넘는 패키지가 나와서 뭔가 수상하다고 느꼈음
  • bash 자동 완성이 점점 “똑똑해지면서” 현재 커서 위치에 파일명이 적절치 않다고 판단하면 파일/디렉토리 완성을 막는 점이 불편함
    • 차라리 항상 파일명 완성으로 fallback하는 게 낫다고 생각함
    • 이 때문에 완성 스크립트를 다 꺼버릴까 고민한 적도 있음
    • bash에는 M-/에 바인딩된 complete-filename처럼 문맥 무시하고 파일명만 완성하는 함수들이 있음
    • 특정 명령어의 파일 완성이 완전히 깨져서, ls로 먼저 완성한 뒤 명령어를 바꾸는 식으로 우회함
    • 파일명이 존재하지만 실행 불가할 때는 “file foo.exe exists but it isn't executable” 같은 메시지를 주는 게 혼란이 덜함
    • complete -r 실행 후 원하는 대로 동작하는 걸 보면 bash-completion 스크립트에 문제가 있는 듯함
    • 웹 폼에서 이메일 입력 시, 첫 글자만 쳐도 “Invalid email!” 오류를 띄우는 프론트엔드 검증처럼 답답한 UX와 비슷함
  • 내가 작성한 글인데, 다른 사람들도 흥미롭게 읽었으면 함
    • zsh 완성 스크립트를 매번 로드하는 대신 $fpath에 설치하면 캐시되어 시작 속도가 빨라짐
    • Homebrew 배포 시 자동으로 completions를 설치할 수 있음
    • zsh 완성은 규모가 커서 익히기 어렵지만, 회사에서 ansible 래퍼 스크립트에 플레이북별 옵션, 태그 자동 완성 등을 추가하며 점점 다듬고 있음
  • 최근 CLI 개발에 usage 라이브러리를 쓰기 시작했음 → clap과 통합되고, completions·argparse·man 페이지 생성 가능
    • fish 스크립트의 argparse 블록을 교체할 가치는 고민 중이지만, optparse보다는 훨씬 나음
    • 나도 비슷한 CLI 사양 정의 도구를 만들고 있는데, fish 지원을 막 추가했음
  • bash/zsh에서 JSON 필드 자동 완성을 지원하는 fx.wtf를 소개함
    • ijq보다 가볍지만, 상황에 따라 유용할 수 있음
  • zsh 내장 함수로 completer를 만드는 튜토리얼 → zsh-completions-howto
  • 프로그램이 bash 스크립트를 작성하지 않아도 되게 하는 표준 플래그가 있는지 궁금함
    • zsh에서는 _gnu_generic으로 --help 기반 간단 완성이 가능함
    • Rust의 clap_complete, ripgrep의 --generate 옵션, PHP Symfony의 런타임 완성 생성 등 사례가 있음
    • 공통 파서 라이브러리들이 자동으로 구현해주는 --completion 표준이 있으면 좋겠음
  • ksh에서는 배열 정의만으로 간단히 완성을 구현할 수 있음
    • 예: complete_kill_1 배열에 시그널 이름을 넣으면 kill 명령 첫 인자 완성 제공
    • 이게 ksh93 문법인지, oksh에 백포트된 건지 궁금함
  • 내가 만든 간단한 zsh 완성 스니펫 예시: set-java-home 함수에서 ~/apps/java/* 목록을 버전으로 완성하도록 _describe 사용
    • 거의 원라인에 가까운 간단한 구조임
  • LLM의 주된 기능이 자동 텍스트 완성이라 그런지, GitHub Copilot이 vscode 터미널에서 bash와 zsh 완성을 꽤 잘해줌