สคริปต์ใน /etc/profile.d ถูกเพิกเฉย?


61

ฉันใหม่กับ Ubuntu ฉันใช้เดสก์ท็อป 13.10

ฉันต้องการตั้งค่านามแฝงทั้งระบบและพร้อมท์แบบกำหนดเองสำหรับการทุบตี ฉันพบบทความนี้:

https://help.ubuntu.com/community/EnvironmentVariables

ทำตามคำแนะนำในบทความนี้ฉันสร้าง /etc/profiles.d/profile_local.sh มันเป็นเจ้าของโดย root และมีสิทธิ์ 644 เหมือนกับสคริปต์อื่น ๆ ที่นั่น:

root@ubuntu:/etc/profile.d# ll
total 28
drwxr-xr-x   2 root root  4096 Mar 23 08:56 .
drwxr-xr-x 135 root root 12288 Mar 23 09:15 ..
-rw-r--r--   1 root root   660 Oct 23  2012 bash_completion.sh
-rw-r--r--   1 root root  3317 Mar 23 07:36 profile_local.sh
-rw-r--r--   1 root root  1947 Nov 23 00:57 vte.sh

ฉันยืนยันเพิ่มเติมว่า / etc / profile calls /etc/profile.d มันมีรหัสบล็อกนี้:

if [ -d /etc/profile.d ]; then
  for i in /etc/profile.d/*.sh; do
    if [ -r $i ]; then
      . $i
    fi
  done
  unset i
fi

เมื่อเข้าสู่ระบบจะไม่ปรากฏว่าสคริปต์ที่กำหนดเอง profile_local.sh ที่ฉันสร้างขึ้นได้รับที่มา อย่างไรก็ตามหากหลังจากเข้าสู่ระบบฉัน 'source /etc.profile.d/profile_local.sh' ฉันจะได้รับพฤติกรรมตามที่คาดหวังชื่อแทนที่กำหนดเองของฉันและพร้อมท์ที่กำหนดเอง

ผมทำอะไรผิดหรือเปล่า?

เนื้อหาของสคริปต์ 'profile_local.sh':

# 3/23/14 - Copied from Gentoo /etc/bash/bashrc
# Placed in /etc/profile.d as described at:
# https://help.ubuntu.com/community/EnvironmentVariables

# This file is sourced by all *interactive* bash shells on startup,
# including some apparently interactive shells such as scp and rcp
# that can't tolerate any output.  So make sure this doesn't display
# anything or bad things will happen !


# Test for an interactive shell.  There is no need to set anything
# past this point for scp and rcp, and it's important to refrain from
# outputting anything in those cases.
if [[ $- != *i* ]] ; then
        # Shell is non-interactive.  Be done now!
        return
fi

# Bash won't get SIGWINCH if another process is in the foreground.
# Enable checkwinsize so that bash will check the terminal size when
# it regains control.  #65623
# http://cnswww.cns.cwru.edu/~chet/bash/FAQ (E11)
shopt -s checkwinsize

# Enable history appending instead of overwriting.  #139609
shopt -s histappend

# Change the window title of X terminals 
case ${TERM} in
        xterm*|rxvt*|Eterm|aterm|kterm|gnome*|interix)
                PROMPT_COMMAND='echo -ne "\033]0;${USER}@${HOSTNAME%%.*}:${PWD/#$HOME/~}\007"'
                ;;
        screen)
                PROMPT_COMMAND='echo -ne "\033_${USER}@${HOSTNAME%%.*}:${PWD/#$HOME/~}\033\\"'
                ;;
esac

use_color=false

# Set colorful PS1 only on colorful terminals.
# dircolors --print-database uses its own built-in database
# instead of using /etc/DIR_COLORS.  Try to use the external file
# first to take advantage of user additions.  Use internal bash
# globbing instead of external grep binary.
safe_term=${TERM//[^[:alnum:]]/?}   # sanitize TERM
match_lhs=""
[[ -f ~/.dir_colors   ]] && match_lhs="${match_lhs}$(<~/.dir_colors)"
[[ -f /etc/DIR_COLORS ]] && match_lhs="${match_lhs}$(</etc/DIR_COLORS)"
[[ -z ${match_lhs}    ]] \
        && type -P dircolors >/dev/null \
        && match_lhs=$(dircolors --print-database)
[[ $'\n'${match_lhs} == *$'\n'"TERM "${safe_term}* ]] && use_color=true

if ${use_color} ; then
        # Enable colors for ls, etc.  Prefer ~/.dir_colors #64489
        if type -P dircolors >/dev/null ; then
                if [[ -f ~/.dir_colors ]] ; then
                        eval $(dircolors -b ~/.dir_colors)
                elif [[ -f /etc/DIR_COLORS ]] ; then
                        eval $(dircolors -b /etc/DIR_COLORS)
                fi
        fi

        if [[ ${EUID} == 0 ]] ; then
                PS1='\[\033[01;31m\]\h\[\033[01;34m\] \W \$\[\033[00m\] '
        else
                PS1='\[\033[01;32m\]\u@\h\[\033[01;34m\] \w \$\[\033[00m\] '
        fi

        alias ls='ls --color=auto'
        alias grep='grep --colour=auto'
else
        if [[ ${EUID} == 0 ]] ; then
                # show root@ when we don't have colors
                PS1='\u@\h \W \$ '
        else
                PS1='\u@\h \w \$ '
        fi
fi

# Try to keep environment pollution down, EPA loves us.
unset use_color safe_term match_lhs

TZ="PST8PDT"

alias ll='ls -la'
alias dig='dig +search'
alias dir='ls -ba'

alias edit="ee"
alias ss="ps -aux"
alias dot='ls .[a-zA-Z0-9_]*'
alias news="xterm -g 80x45 -e trn -e -S1 -N &"

alias more="less"
alias c="clear"
alias m="more"
alias j="jobs"

# common misspellings
alias mroe=more
alias pdw=pwd

1
ไม่มันไม่สามารถเรียกใช้งานได้ แต่ไม่มีอีกสองสคริปต์ อย่างไรก็ตามฉันเปลี่ยนและลองใหม่อีกครั้ง ยังไม่มีโชค
Drew

3
สิ่งนี้ไม่มีส่วนเกี่ยวข้องกับการเพิ่ม.shมันไม่เกี่ยวข้องและไฟล์profile.dที่มานั้นไม่ได้ดำเนินการซึ่งแตกต่างกันเล็กน้อยและไม่ต้องการให้ไฟล์นั้นสามารถเรียกใช้งานได้ ปัญหาที่นี่คือว่าprofile& co ไม่ได้อ่านโดยสคริปต์ที่ไม่ใช่การเข้าสู่ระบบ
terdon

1
ดึงอ่านคำตอบของฉัน ไฟล์โปรไฟล์ถูกละเว้นโดยเชลล์ที่ไม่ใช่การเข้าสู่ระบบ แต่ล็อกอิน GUI เริ่มต้นของ Ubuntu จะอ่านบางไฟล์ เพียงแค่ใช้.bashrcและปัญหาทั้งหมดของคุณจะหายไป นอกจากนี้ยังมีคำถามเกี่ยวกับการมาก่อนถ้าหนึ่งในไฟล์ที่อ่านในภายหลังตั้งค่า PS1 แล้วค่าก่อนหน้านี้จะถูกยกเลิก อย่างไรก็ตามอย่างจริงจังอย่าแตะต้อง filers ใน/etcเล่นกับคนในบ้าน dir ของคุณและใช้.bashrcโปรไฟล์ไม่
terdon

1
ใช่นั่นควรจะเป็นเชลล์การเข้าสู่ระบบ (นั่นคือสิ่งที่ควรรวมไว้ในคำถามของคุณในครั้งต่อไป) แต่ระบบส่วนใหญ่จะมีการเริ่มต้นไฟล์ในบ้านของคุณและการตั้งค่าจะมีเขียนทับสิ่งที่คุณทำใน.profile /etc/profileโดยทั่วไปไม่เคยสัมผัส/etcเว้นแต่คุณรู้ว่าคุณกำลังทำอะไร นั่นคือสิ่งที่ไฟล์สำหรับผู้ใช้เฉพาะเจาะจง นอกจากนี้โปรดแก้ไขคำถามของคุณและอธิบายอย่างชัดเจนว่าคุณกำลังเชื่อมต่ออย่างไรซึ่งเปลี่ยนแปลงทุกอย่าง
terdon

4
โปรดอย่าใช้สิ่งนี้/etc/profile.dซึ่งเป็นแนวคิดที่ไม่ดีจริงๆและจะส่งผลกระทบต่อผู้ใช้ทุกคนของระบบ มีเพียงคำสั่งจากprofile_local.shในของคุณ~/.profileหรือเพียงแหล่งสคริปต์โดยการเพิ่มบรรทัดนี้ของคุณ:~/.profile . /path/to/profile_local.sh( .หมายถึงsourceมันจะอ่านไฟล์ที่คุณให้และเรียกใช้คำสั่งที่พบที่นั่น)
terdon

คำตอบ:


108

เพื่อให้เข้าใจสิ่งที่เกิดขึ้นที่นี่คุณต้องเข้าใจข้อมูลเบื้องหลังเล็กน้อยเกี่ยวกับวิธีการเรียกใช้เชลล์ (ทุบตีในกรณีนี้)

  • เมื่อคุณเปิดเทอร์มินัลอีมูเลเตอร์ ( gnome-terminalตัวอย่าง) คุณกำลังดำเนินการสิ่งที่เรียกว่าเชลล์แบบโต้ตอบและไม่ใช่การล็อกอิน

  • เมื่อคุณล็อกอินเข้าสู่เครื่องจากบรรทัดคำสั่งผ่านsshหรือเรียกใช้คำสั่งเช่นsu - usernameคุณกำลังเรียกใช้เชลล์ล็อกอินแบบโต้ตอบ

  • เมื่อคุณเข้าสู่ระบบแบบกราฟิกคุณกำลังเรียกใช้บางสิ่งที่แตกต่างอย่างสิ้นเชิงรายละเอียดจะขึ้นอยู่กับระบบและสภาพแวดล้อมแบบกราฟิกของคุณ แต่โดยทั่วไปแล้วมันคือกราฟิกเชลล์ที่เกี่ยวข้องกับการเข้าสู่ระบบของคุณ ในขณะที่เชลล์กราฟิกจำนวนมาก (รวมถึงค่าเริ่มต้นของ Ubuntu) จะอ่าน/etc/profileไม่ได้ทั้งหมด

  • ในที่สุดเมื่อคุณเรียกใช้สคริปต์เปลือกก็จะทำงานในไม่โต้ตอบเปลือกที่ไม่ได้เข้าสู่ระบบ

ตอนนี้ไฟล์ที่ bash จะอ่านเมื่อเรียกใช้ขึ้นอยู่กับชนิดของเชลล์ที่มันรันอยู่ ต่อไปนี้เป็นข้อความที่ตัดตอนมาจากส่วน INVOCATION ของman bash(เหมืองของการเน้น):

เมื่อ bash ถูกเรียกใช้เป็นเชลล์ล็อกอินแบบโต้ตอบหรือเป็นเชลล์ที่ไม่มีการโต้ตอบกับตัวเลือก --login มันจะอ่านและเรียกใช้คำสั่งจากไฟล์/ etc / profileก่อนหากไฟล์นั้นมีอยู่ หลังจากอ่านไฟล์นั้นจะค้นหา~ / .bash_profile, ~ / .bash_login และ ~ / .profile ตามลำดับและอ่านและเรียกใช้คำสั่งจากไฟล์แรกที่มีอยู่และอ่านได้ ตัวเลือก --noprofile อาจถูกใช้เมื่อเชลล์เริ่มต้นเพื่อยับยั้งพฤติกรรมนี้

เมื่อเชลล์เชิงโต้ตอบที่ไม่ใช่เชลล์ล็อกอิน เริ่มทำงาน bash จะอ่านและดำเนินการคำสั่งจาก/etc/bash.bashrcและ~ / .bashrcหากมีไฟล์เหล่านี้อยู่ สิ่งนี้อาจถูกยับยั้งโดยใช้ตัวเลือก --norc ตัวเลือกไฟล์ --rcfile จะบังคับให้ทุบตีเพื่ออ่านและดำเนินการคำสั่งจากไฟล์แทน /etc/bash.bashrc และ ~ / .bashrc

หมายความว่าคุณแก้ไขไฟล์ผิด คุณสามารถทดสอบสิ่งนี้ได้โดยการวางไปที่คอนโซลเสมือนโดยใช้Ctrl+ Alt+ F2(กลับไปที่ GUI ด้วยAlt+ F7หรือF8ขึ้นอยู่กับการตั้งค่าของคุณ) และเข้าสู่ระบบที่นั่น คุณจะเห็นว่าพรอมต์และชื่อแทนของคุณพร้อมใช้งาน

ดังนั้นเพื่อให้การตั้งค่าที่คุณต้องการนำไปใช้กับเชลล์ที่ไม่ใช่การเข้าสู่ระบบประเภทที่คุณได้รับทุกครั้งที่คุณเปิดเทอร์มินัลคุณควรทำการเปลี่ยนแปลง~/.bashrcแทน หรือคุณสามารถวางนามแฝงไว้ในไฟล์ได้~/.bash_aliases(โปรดทราบว่านี่เป็นคุณสมบัติของอูบุนตูและคุณไม่ควรคาดหวังให้มันทำงานบนดิสทริบิวชันอื่น ๆ )

สำหรับรายละเอียดเพิ่มเติมเกี่ยวกับไฟล์ที่ควรจะใช้สำหรับสิ่งที่เห็นที่นี่


หมายเหตุ:

  • Debian (และนามสกุลอูบุนตู) นอกจากนี้ยังมีการเริ่มต้นที่มา~/.profile ~/.bashrcซึ่งหมายความว่าการเปลี่ยนแปลงใด ๆ ที่คุณทำ~/.bashrcจะได้รับสืบทอดจากล็อกอินเชลล์ แต่ i) กรณีนี้ไม่ได้เกิดขึ้นในเครื่อง Linux / Unix และ ii) การผกผันไม่เป็นความจริงซึ่งเป็นเหตุให้คุณมักจะทำงานร่วมกับ~/.bashrc& co มากกว่า~/.profileหรือ/etc/profile.

  • นอกจากนี้บันทึกทั่วไปเกี่ยวกับการใช้งานการเปลี่ยนแปลงที่ทำกับไฟล์การกำหนดค่าใน/etcจะมีผลกับผู้ใช้ทั้งหมด นี่ไม่ใช่สิ่งที่คุณต้องการทำและควรหลีกเลี่ยง คุณควรใช้ไฟล์ที่เทียบเท่าในไดเรกทอรีบ้านของคุณ ( ~/)

  • ไฟล์การกำหนดค่าต่างๆจะอ่านตามลำดับ สำหรับเชลล์ล็อกอินโดยเฉพาะลำดับคือ:

    /etc/profile -> /etc/profile.d/* (in alphabetical order) -> ~/.profile

    ซึ่งหมายความว่าการตั้งค่าใด ๆ~/.profileจะเขียนทับสิ่งที่ตั้งค่าไว้ในไฟล์ก่อนหน้า


1
ตามโพสต์นี้howtolamp.com/articles/ …คุณยังสามารถเรียกใช้echo $0จากเทอร์มินัลและถ้าเอาท์พุทจะนำหน้าด้วย "-" แล้วคุณจะอยู่ในเปลือกเข้าสู่ระบบ
stackoverflower

@stackoverflower ไม่ได้อยู่ในระบบของฉัน ใช้งานได้กับเชลล์ล็อกอินแบบโต้ตอบระยะไกล bash -lดูเหมือนจะไม่ได้เมื่อทำงาน ไม่ว่าในกรณีใดเหตุใดจึงมีความเกี่ยวข้อง คำถามไม่ได้เกี่ยวกับวิธีตรวจสอบว่าคุณใช้เชลล์ประเภทใด
terdon

โพสต์ยอดเยี่ยม, สงสัยว่าทำไมมันไม่ได้แสดงใน google เมื่อฉันมีปัญหาเดียวกัน
Donato

@stackoverflower หาก"$0"ขยายไปยังสิ่งที่ขึ้นต้นด้วย-คุณจะรู้ว่าคุณมีเปลือกเข้าสู่ระบบ แต่การสนทนานั้นไม่เป็นความจริง: การไม่มีตัวตน-ไม่แน่ใจว่าคุณไม่ได้อยู่ในเชลล์การเข้าสู่ระบบ วิธีที่พบได้บ่อยที่สุดในการเริ่มต้นเชลล์การเข้าสู่ระบบจะทำให้คุณเป็นผู้นำ-แต่ไม่ใช่ทั้งหมด man bashบอกเราว่า" เชลล์การเข้าสู่ระบบเป็นหนึ่งที่มีอักขระตัวแรกของอาร์กิวเมนต์ศูนย์เป็น-หรือเริ่มต้นด้วย--loginตัวเลือก" ( -lเป็นรูปแบบย่อของ--login; เทียบเท่ากัน) ใน Bash คุณสามารถเรียกใช้shopt login_shellเพื่อตรวจสอบได้
Eliah Kagan

2

เป็นไปได้อีกโดยเฉพาะอย่างยิ่งสำหรับการตั้งค่าเช่นการตั้งค่าประวัติHISTSIZE, HISTFILESIZE, HISTCONTROLและPS1เป็นว่าไฟล์ที่กำลังโหลด ~/.bashrcแต่การตั้งค่าจะถูกเขียนทับในแฟ้มอื่นที่เป็นแหล่งที่มาต่อมามีการกระทำผิดส่วนใหญ่มีแนวโน้มเป็น (ฉันมีชุดการตั้งค่าเริ่มต้นสำหรับเซิร์ฟเวอร์ของเราเช่นพรอมต์ที่เป็นสีแดงสำหรับรูตเพื่อเตือนผู้ใช้และประวัติขนาดใหญ่ที่มีการประทับเวลา)

เริ่มต้นอูบุนตู.bashrcจาก/etc/skelชุดการตั้งค่าหลายอย่างซึ่งอาจจะทำให้ความรู้สึกในการตั้งค่าจากที่ไหนสักแห่งที่มันจะไม่แทนที่การตั้งค่าที่กำหนดโดยเจ้าของระบบจาก/etc/profile.d(Like /etc/bash.bashrc) (หากผู้ใช้แก้ไขของพวกเขา.bashrcมันเป็นเรื่องดีที่จะเขียนทับการตั้งค่าไฟล์เริ่มต้นระบบ น่ารำคาญกว่า)


2

ในเซสชัน Debian for Terminal ฉันแก้ไขปัญหานี้ให้กับผู้ใช้ทั้งหมดดังนี้

เพิ่มไปยัง

sudo nano /etc/bash.bashrc

บล็อก

if [ -d /etc/profile.d ]; then
  for i in /etc/profile.d/*.sh; do
    if [ -r $i ]; then
      . $i
    fi
  done
  unset i
fi

จาก

/etc/profile

1

ตามเส้นทางนี้:

  • เปิดแก้ไข -> การตั้งค่า
  • ในแท็บแรก "ทั่วไป" ภายใต้ป้ายกำกับ "คำสั่ง" ให้เปิดใช้งาน "เรียกใช้คำสั่งเป็นล็อกอินเชลล์"

-1

VERSION = "16.04.3 LTS (Xenial Xerus)"

โอเคดังนั้นทุกคนจึงสันนิษฐานว่าคนที่นี่ไม่ต้องการ /etc/profile.d/somefile.sh สำหรับผู้ใช้ทั้งหมด แต่ในกรณีของฉันนั่นคือสิ่งที่ฉันต้องการอย่างแท้จริง

ดังนั้นจริงๆแล้วมันจะเปิดออกด้วย Ubuntu หากคุณใช้สิ่งนี้และคุณต้องการให้มันมีผลในเชลล์กราฟิกของคุณสิ่งที่คุณต้องทำคือตั้งค่าไฟล์แล้วออกจากระบบและกลับมาใหม่อีกครั้ง คอนโซลทั้งหมดของคุณหรือสิ่งที่คุณเรียกใช้ไม่ว่าจะเป็นชนิด xterm หรือประเภทคอนโซล (หรือวางลงในเชลล์) จะมีไฟล์ดังกล่าวมา

ไม่จำเป็นต้องใช้. bashrc ฯลฯ สำหรับผู้ใช้ทั้งหมด ขออภัยที่ไม่ชัดเจนในคำตอบข้างต้น ทุกสิ่งที่พวกเขากล่าวว่าเป็นจริง แต่ในความเป็นจริงมันไม่เป็นความจริงเพราะทุกอย่างที่ windows manager เปิดตัวจะสืบทอดการตั้งค่าเหล่านี้ดังนั้นเพียงแค่ลงชื่อเข้าใช้ใหม่และแก้ไขปัญหาของคุณและไม่ต้องกังวลกับ. bashrc .


2
ปัญหาของฉันคือสิ่งนี้ไม่ได้เกิดขึ้นใน terminal ที่ทำงานภายใต้ส่วนต่อประสานกราฟิกกับผู้ใช้ ไม่ได้อยู่ใน Ubuntu 16.04.3 หรือ 18.04
Thomas Arildsen
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.