Giriş yaptıktan sonra “t♦: komut bulunamadı”


24

Lubuntu'yu 12.10'dan 13.04'e yükselttikten sonra bu sorunu yaşıyorum.

Ben basın Ctrl+ Alt+ 1, giriş, şifre girmek iki saniye bekleyin ve olsun: ♦: command not found". Bu mesajdan sonra sorunsuz komutlar yazabilirim, fakat nedir?

echo $PATH
/usr/lib/lightdm/lightdm:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games:/home/vitaly/bin:/usr/java/jdk1.7.0_17/bin

Benim .bashrcdosyam:

# ~/.bashrc: executed by bash(1) for non-login shells.
# see /usr/share/doc/bash/examples/startup-files (in the package bash-doc)
# for examples

# If not running interactively, don't do anything
case $- in
    *i*) ;;
      *) return;;
esac

# don't put duplicate lines or lines starting with space in the history.
# See bash(1) for more options
HISTCONTROL=ignoreboth

# append to the history file, don't overwrite it
shopt -s histappend

# for setting history length see HISTSIZE and HISTFILESIZE in bash(1)
HISTSIZE=1000
HISTFILESIZE=2000

# check the window size after each command and, if necessary,
# update the values of LINES and COLUMNS.
shopt -s checkwinsize

# If set, the pattern "**" used in a pathname expansion context will
# match all files and zero or more directories and subdirectories.
#shopt -s globstar

# make less more friendly for non-text input files, see lesspipe(1)
[ -x /usr/bin/lesspipe ] && eval "$(SHELL=/bin/sh lesspipe)"

# set variable identifying the chroot you work in (used in the prompt below)
if [ -z "${debian_chroot:-}" ] && [ -r /etc/debian_chroot ]; then
    debian_chroot=$(cat /etc/debian_chroot)
fi

# set a fancy prompt (non-color, unless we know we "want" color)
case "$TERM" in
    xterm-color) color_prompt=yes;;
esac

# uncomment for a colored prompt, if the terminal has the capability; turned
# off by default to not distract the user: the focus in a terminal window
# should be on the output of commands, not on the prompt
#force_color_prompt=yes

if [ -n "$force_color_prompt" ]; then
    if [ -x /usr/bin/tput ] && tput setaf 1 >&/dev/null; then
    # We have color support; assume it's compliant with Ecma-48
    # (ISO/IEC-6429). (Lack of such support is extremely rare, and such
    # a case would tend to support setf rather than setaf.)
    color_prompt=yes
    else
    color_prompt=
    fi
fi

if [ "$color_prompt" = yes ]; then
    PS1='${debian_chroot:+($debian_chroot)}\[\033[01;32m\]\u@\h\[\033[00m\]:\[\033[01;34m\]\w\[\033[00m\]\$ '
else
    PS1='${debian_chroot:+($debian_chroot)}\u@\h:\w\$ '
fi
unset color_prompt force_color_prompt

# If this is an xterm set the title to user@host:dir
case "$TERM" in
xterm*|rxvt*)
    PS1="\[\e]0;${debian_chroot:+($debian_chroot)}\u@\h: \w\a\]$PS1"
    ;;
*)
    ;;
esac

# enable color support of ls and also add handy aliases
if [ -x /usr/bin/dircolors ]; then
    test -r ~/.dircolors && eval "$(dircolors -b ~/.dircolors)" || eval "$(dircolors -b)"
    alias ls='ls --color=auto'
    #alias dir='dir --color=auto'
    #alias vdir='vdir --color=auto'

    alias grep='grep --color=auto'
    alias fgrep='fgrep --color=auto'
    alias egrep='egrep --color=auto'
fi

# some more ls aliases
alias ll='ls -alF'
alias la='ls -A'
alias l='ls -CF'

# Add an "alert" alias for long running commands.  Use like so:
#   sleep 10; alert
alias alert='notify-send --urgency=low -i "$([ $? = 0 ] && echo terminal || echo error)" "$(history|tail -n1|sed -e '\''s/^\s*[0-9]\+\s*//;s/[;&|]\s*alert$//'\'')"'

# Alias definitions.
# You may want to put all your additions into a separate file like
# ~/.bash_aliases, instead of adding them here directly.
# See /usr/share/doc/bash-doc/examples in the bash-doc package.

if [ -f ~/.bash_aliases ]; then
    . ~/.bash_aliases
fi

# enable programmable completion features (you don't need to enable
# this, if it's already enabled in /etc/bash.bashrc and /etc/profile
# sources /etc/bash.bashrc).
if ! shopt -oq posix; then
  if [ -f /usr/share/bash-completion/bash_completion ]; then
    . /usr/share/bash-completion/bash_completion
  elif [ -f /etc/bash_completion ]; then
    . /etc/bash_completion
  fi
fi

Benim .profiledosyam:

# ~/.profile: executed by the command interpreter for login shells.
# This file is not read by bash(1), if ~/.bash_profile or ~/.bash_login
# exists.
# see /usr/share/doc/bash/examples/startup-files for examples.
# the files are located in the bash-doc package.

# the default umask is set in /etc/profile; for setting the umask
# for ssh logins, install and configure the libpam-umask package.
#umask 022

# if running bash
if [ -n "$BASH_VERSION" ]; then
    # include .bashrc if it exists
    if [ -f "$HOME/.bashrc" ]; then
    . "$HOME/.bashrc"
    fi
fi

# set PATH so it includes user's private bin if it exists
if [ -d "$HOME/bin" ] ; then
    PATH="$HOME/bin:$PATH"
fi

Dosya /etc/profileburada: http://paste.ubuntu.com/5781361/


1
Çıktı nedir echo $PATH? (Lütfen sorunuzu düzenleyin ve ardından cevaplayın).
Seth

1
~ / .Bashrc ve ~ / .profile dosyalarınızı paste.ubuntu.com adresine yükleyip linkleri gönderebilir misiniz?
Eric Carvalho

Gelecekte referans olması için, şekle "elmas" denir.
user98085

echo $PATHSoruma eklendi .
Vitaly Zdanevich

'.Bashrc' dizininde yol değişkeniniz etrafında tek tırnak kullanırsanız, bu soruna neden olabilir.
phyatt

Yanıtlar:


28

Geçici çözüm

İlk önce, tty1 - Ctrl+ Alt+ ' ya girdiğiniz zaman bahsettiğinizi düşünüyorum F1.

Şimdi, bence en büyük olasılıkla söylediğiniz şey oluyor çünkü because ( elmas takım elbise karakteri veya askubuntu moderatörleri için özel bir rozet) gibi garip bir karakteriniz var ~/.bashrcya da ~/.profiledosya içinde ya da çeşitli başlatma komutları içeren başka bir dosya.

Bir sonraki resimde gördüğünüz gibi, ~/.bashrc♦ karakterini tek bir satıra yerleştiren dosyayı düzenledim . Sonuç olarak, terminali açtığımda, sizin tarafınızdan tarif edilen problemi alıyorum:

terminal

Birlikte tty1 gittiğinizde aynı oluyor Ctrl+ Alt+ F1.

Bir kabuk çağrıldığında başlatma komutları içeren dosyalar: /etc/profile, /etc/bashrc, ~/.bash_login, ~/.profile, ~/.bashrc, ~/.bash_aliasesve belki de diğerleri. Bkz. Shell başlatma dosyaları .

Bu dosyalardan birinin içinde yanlış bir şey olup olmadığını hızlı bir şekilde kontrol etmek için sourcekomutu kullanabilirsiniz . Örneğin:

source ~/.bashrc

kaynak profili

Son çözüm

Teftiş sonra /etc/profilegelen http://paste.ubuntu.com/5781361/ , ben hat 31 üzerinde olduğunu tespit - "Sağdan Sola geçersiz kılma"‮ Unicode karakter. Sadece /etc/profiledosyayı açın sudo -H gedit /etc/profile, bu garip karakteri sildiğinizden emin olun; sorun kaybolur.

profil dosyası

Eğlence olarak, örneğin HTML'de, bu unicode karakteri ‮bir satırın önüne ondalık kod kullanarak eklerseniz , neler olduğuna bakın:

Bu yazı Arapça-İngilizce!

Başka bir genelleştirilmiş çözüm

Bir " tuzak " kullanarak hataya neden olan tam komutu bulacağız .

Öncelikle, ~/bindizinde yeni bir komut dosyası yapmalıyız , hadi lib.trap.sh( gedit ~/bin/lib.trap.sh) ile içini şöyle çağıralım :

lib_name='trap'

lib_version=20130620
#changed from lib_version=20121026 found it at /programming//a/13099228/2353900 to work well at initialization of the shell

stderr_log="/dev/shm/stderr.log"

#
# TO BE SOURCED ONLY ONCE:
#
###~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~##

if test "${g_libs[$lib_name]+_}"; then
    return 0
else
    if test ${#g_libs[@]} == 0; then
        declare -A g_libs
    fi
    g_libs[$lib_name]=$lib_version
fi


#
# MAIN CODE:
#
###~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~##

set -o pipefail  # trace ERR through pipes
set -o errtrace  # trace ERR through 'time command' and other functions
set -o nounset   ## set -u : exit the script if you try to use an uninitialised variable
set -o errexit   ## set -e : exit the script if any statement returns a non-true return value

exec 2>"$stderr_log"


###~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~##
#
# FUNCTION: EXIT_HANDLER
#
###~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~##

function exit_handler ()
{
    local error_code="$?"

    test $error_code == 0 && return;

    #
    # LOCAL VARIABLES:
    # ------------------------------------------------------------------
    #    
    local i=0
    local regex=''
    local mem=''

    local error_file=''
    local error_lineno=''
    local error_message='unknown'

    local lineno=''


    #
    # PRINT THE HEADER:
    # ------------------------------------------------------------------
    #
    # Color the output if it's an interactive terminal
    test -t 1 && tput bold; tput setf 4                                 ## red bold
    echo -e "\n(!) EXIT HANDLER\n"


    #
    # GETTING LAST ERROR OCCURRED:
    # ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ #

    #
    # Read last file from the error log
    # ------------------------------------------------------------------
    #
    if test -f "$stderr_log"
        then
            stderr=$( tail -n 1 "$stderr_log" )
            rm "$stderr_log"
    fi

    #
    # Managing the line to extract information:
    # ------------------------------------------------------------------
    #

    if test -n "$stderr"
        then        
            # Exploding stderr on :
            mem="$IFS"
            local shrunk_stderr=$( echo "$stderr" | sed 's/\: /\:/g' )
            IFS=':'
            local stderr_parts=( $shrunk_stderr )
            IFS="$mem"

            # Storing information on the error
            error_file="${stderr_parts[0]}"
            error_lineno="${stderr_parts[1]}"
            error_message=""

            for (( i = 3; i <= ${#stderr_parts[@]}; i++ ))
                do
                    error_message="$error_message "${stderr_parts[$i-1]}": "
            done

            # Removing last ':' (colon character)
            error_message="${error_message%:*}"

            # Trim
            error_message="$( echo "$error_message" | sed -e 's/^[ \t]*//' | sed -e 's/[ \t]*$//' )"
    fi

    #
    # GETTING BACKTRACE:
    # ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ #
    _backtrace=$( backtrace 2 )


    #
    # MANAGING THE OUTPUT:
    # ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ #

    local lineno=""
    regex='^([a-z]{1,}) ([0-9]{1,})$'

    if [[ $error_lineno =~ $regex ]]

        # The error line was found on the log
        # (e.g. type 'ff' without quotes wherever)
        # --------------------------------------------------------------
        then
            local row="${BASH_REMATCH[1]}"
            lineno="${BASH_REMATCH[2]}"

            echo -e "FILE:\t\t${error_file}"
            echo -e "${row^^}:\t\t${lineno}\n"

            echo -e "ERROR CODE:\t${error_code}"             
            test -t 1 && tput setf 6                                    ## white yellow
            echo -e "ERROR MESSAGE:\n$error_message"


        else
            regex="^${error_file}\$|^${error_file}\s+|\s+${error_file}\s+|\s+${error_file}\$"
            if [[ "$_backtrace" =~ $regex ]]

                # The file was found on the log but not the error line
                # (could not reproduce this case so far)
                # ------------------------------------------------------
                then
                    echo -e "FILE:\t\t$error_file"
                    echo -e "ROW:\t\tunknown\n"

                    echo -e "ERROR CODE:\t${error_code}"
                    test -t 1 && tput setf 6                            ## white yellow
                    echo -e "ERROR MESSAGE:\n${stderr}"

                # Neither the error line nor the error file was found on the log
                # (e.g. type 'cp ffd fdf' without quotes wherever)
                # ------------------------------------------------------
                else
                    #
                    # The error file is the first on backtrace list:

                    # Exploding backtrace on newlines
                    mem=$IFS
                    IFS='
                    '
                    #
                    # Substring: I keep only the carriage return
                    # (others needed only for tabbing purpose)
                    IFS=${IFS:0:1}
                    local lines=( $_backtrace )

                    IFS=$mem

                    error_file=""

                    if test -n "${lines[1]}"
                        then
                            array=( ${lines[1]} )

                            for (( i=2; i<${#array[@]}; i++ ))
                                do
                                    error_file="$error_file ${array[$i]}"
                            done

                            # Trim
                            error_file="$( echo "$error_file" | sed -e 's/^[ \t]*//' | sed -e 's/[ \t]*$//' )"
                    fi

            echo -e "ROW, FILE:\t\t${lines[2]   }\n"

                    echo -e "ERROR CODE:\t${error_code}"
                    test -t 1 && tput setf 6                            ## white yellow
                    if test -n "${stderr}"
                        then
                            echo -e "ERROR MESSAGE:\n${stderr}"
                        else
                            echo -e "ERROR MESSAGE:\n${error_message}"
                    fi
            fi
    fi

    #
    # PRINTING THE BACKTRACE:
    # ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ #

    test -t 1 && tput setf 7                                            ## white bold
    echo -e "\n$_backtrace\n"

    #
    # EXITING:
    # ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ #

    test -t 1 && tput setf 4                                            ## red bold
    echo "Exiting!"

    test -t 1 && tput sgr0 # Reset terminal

    exit "$error_code"
}
trap exit_handler ERR                                                  # ! ! ! TRAP EXIT ! ! !
#trap exit ERR                                                        # ! ! ! TRAP ERR ! ! ! 


###~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~##
#
# FUNCTION: BACKTRACE
#
###~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~##

function backtrace
{
    local _start_from_=0

    local params=( "$@" )
    if (( "${#params[@]}" >= "1" ))
        then
            _start_from_="$1"
    fi

    local i=0
    local first=false
    while caller $i > /dev/null
    do
        if test -n "$_start_from_" && (( "$i" + 1   >= "$_start_from_" ))
            then
                if test "$first" == false
                    then
                        echo "BACKTRACE IS:"
                        first=true
                fi
                caller $i
        fi
        let "i=i+1"
    done
}

return 0

Şimdi, yapmanız gereken tek şey dosyanın başında bir sonraki satır koymaktır /etc/profile( sudo -H gedit /etc/profile):

source '/home/<user_name>/bin/lib.trap.sh'

<user_name>Kullanıcı adınızla değiştirin . Bu şekilde, bir kabuk çağrıldığında başlatma komutları içeren tüm dosyalar "tuzak" içinden geçecektir.

/etc/profileÖrneğin yanlış bir komut olup olmadığını test etmek için, terminal sonraki komutlarında çalıştırın:

bash kaynağı / etc / profile

Bir şey yanlışsa, bu durumda olduğu gibi, sonuç şöyle olacaktır:

tuzak

Bu nedenle, şimdi satırda dosyada bir sorun ( command not found) olduğundan emin olduğumuzu biliyoruz ( /etc/profileyukarıdaki satırda 31 değil çünkü dosya başında yeni bir satır ekledik).

Luca Borrione'ye , bu genelleştirilmiş çözümü tamamlamama yardımcı olan bu cevabındaki senaryosu için çok teşekkür ederim .


'Den .bashrcve .profilesoruya kod eklendi - Burada ♦ bulamıyorum.
Vitaly Zdanevich

1
Sen başlatma komutları (içeren tüm dosyaları kontrol etmelidir @VitalyZdanevich .bash_aliases, .pam_environment, mutlaka tam olarak bu karakteri garip içini somenthing için, vs).
Radu Rădeanu

2
Evet, sorun 31. satırda /etc/profile. Orada çok garip bir şey var. Sadece herşeyi silin five JAVA_HOMEbir veya iki 'Enter' tuşuna bastıktan sonra her şey yolunda olmalı. Dosyasıylasudo gedit /etc/profile
Radu Rădeanu

1
@RyanLoremIpsum Hayır, eminim. OP oraya yapıştırdı, ben değil :)
Radu Rădeanu

1
Harika! Bu iki şeffaf çizgiyi temizledim /etc/profile. Garip çanta
Vitaly Zdanevich,

5

Bash'nin başlatma komut dosyalarının hatalarını ayıklamak için aşağıdakileri çalıştırın (sanal konsolda oturum açtıktan sonra).

PS4='+ $BASH_SOURCE:$LINENO:' bash -xlic ''

Yukarıdakiler bash (interaktif ( -i) login ( -l) modunda çalışır, loginsanal bir konsola girdiğinizde programın yaptığı gibi . -c ''başlatma komut dosyalarını çalıştırdıktan hemen sonra çıkmasını sağlar ve -xve PS4=...komutları çalıştırmadan önce her komutun dosya adını ve bu komutun satır numarasını yazmasını sağlar. Bu, geçersiz komutun hangi dosyanın hangi satırı bulunduğunu belirlemeye yardımcı olacaktır.

Yandaki notta ♦, sanal konsol için varsayılan fontun, sembolünün bulunmadığı karakterleri yazdırmak için kullandığı semboldür.


1

İlklendirme dosyalarınızı ararken, ♦ çıktısı için kullanılan onaltılık sayıyı aramak yararlı olabilir. Unicode Character 'BLACK DIAMOND SUIT' e göre for için hex kodu 2666'dır . Not: Aynı veya benzer görünümlü sembol üreten en az bir tane hex hex kodu (25C6) vardır. "Elmas" için arama sonuçlarına bakın. Unicode Karakter Arama

Belki de bir şey \u2666senaryolardan birinde. Gönderen yankı Kılavuzu Bash Referans - "\ uhhhh Unicode (ISO / IEC 10646) değeri onaltılık değer HHHH olan karakteri (bir ila dört basamaklı hex)"

Kullanılan karakter kodlamasına bağlıdır, bu yüzden önce en muhtemel olanları aramak isteyebilirsiniz. echo $LC_CTYPEkabuğunuz tarafından kullanılan karakter kodlamasını döndürmelidir. Terminalin Karakter Kodlamasını Alma bölümüne bakın.


-1 çünkü ~/.bash_historyPS1'de etkileşimli olarak çalıştırılan komutlar saklanır.
Radu Rădeanu

Onaylamak için teşekkür ederim. Kaldırdım. Gerisini cevabınıza yorum olarak eklemeli miydim, Radu?
iyrin

Oh, anlamıyorum - ne yapmam gerekiyor? Lubuntu başlatma dosyalarının neresinde? Tam metin aramayı \u2666ve yayınlamayı Catfish'te (Lubuntu search) denedim - hiçbir şey. Ekim history- hiçbir şey. Bu mesajı yalnızca oturum açtıktan sonra yalnızca tty olarak görüyorum. echo $LC_CTYPEBoş satırdan sonra .
Vitaly Zdanevich

Çalışan localeLC_CTYPE göstermelidir. yerel ayar
iyrin

Bundan önce Radu'nun cevabını dene! Tty'nizde kullanılan karakter kümesini daraltırsak, katı elmas için ilgili karakter kodunun oluşumunu arayabilirsiniz. Radu’nun RLO’nun bulduğu sonuç nedense, bu araştırma sonuçsuz kalacaktır.
iyrin

0

Tam yolunu, bashrc dosyanızı tam yolun yanı sıra bashrc dosyanızı düzenlemenizi sağlayacak bir araç yazın.

/bin/nano /home/username/.bashrc

PATHDeğişkeni kötüye kullanma bul ve yorumla. Büyük olasılıkla, yolunuza bir şey eklemeye çalışırken, çift alıntı yerine tek alıntı yapıldı.

PATH='$PATH:/path/to/new/tool' # very BAD, single quotes won't expand PATH
#    ^                       ^

PATH="$PATH:/path/to/new/tool" # Good! The double quotes allow variable expansion

Bash kullanımıyla ilgili göze çarpan herhangi bir sorun olup olmadığını görmek için .bashrc dosyasını https://www.shellcheck.net/ gibi bir araca kopyalayın .

Umarım yardımcı olur.

Sitemizi kullandığınızda şunları okuyup anladığınızı kabul etmiş olursunuz: Çerez Politikası ve Gizlilik Politikası.
Licensed under cc by-sa 3.0 with attribution required.