답변:
간단하게 유지하십시오 :)
grep + echo로 충분합니다 :
grep -qxF 'include "/configs/projectname.conf"' foo.bar || echo 'include "/configs/projectname.conf"' >> foo.bar
-q
조용히 해-x
전체 라인과 일치-F
패턴은 일반 문자열입니다편집 : 통합 @cerin 및 @ thijs-wouters 제안 .
grep -vq ...
echo
줄의 cat
heredoc을 구성 파일로 트리거 합니다.
-s
파일이 존재하지 않을 때 오류를 무시하고 해당 줄만있는 새 파일을 작성하려면 추가하십시오 .
이것은 사용하여 깨끗하고 읽기 쉽고 재사용 가능한 솔루션을 것 grep
하고 echo
이미 존재하지 않는 경우에만 파일에 줄을 추가 :
LINE='include "/configs/projectname.conf"'
FILE='lighttpd.conf'
grep -qF -- "$LINE" "$FILE" || echo "$LINE" >> "$FILE"
전체 라인 사용과 일치해야하는 경우 grep -xqF
-s
파일이 존재하지 않을 때 오류를 무시하고 해당 줄만있는 새 파일을 작성하려면 추가하십시오 .
sudo grep -qF "$LINE" "$FILE" || echo "$LINE" | sudo tee -a "$FILE"
-
.
--
grep 명령을 추가 했으므로 대시로 시작하는 $ LINE을 더 이상 옵션으로 해석하지 않습니다.
sed
버전은 다음과 같습니다 .
sed -e '\|include "/configs/projectname.conf"|h; ${x;s/incl//;{g;t};a\' -e 'include "/configs/projectname.conf"' -e '}' file
문자열이 변수에있는 경우 :
string='include "/configs/projectname.conf"'
sed -e "\|$string|h; \${x;s|$string||;{g;t};a\\" -e "$string" -e "}" file
sed -i -e '\|session.*pam_mkhomedir.so|h; ${x;s/mkhomedir//;{g;tF};a\' -e 'session\trequired\tpam_mkhomedir.so umask=0022' -e '};:F;s/.*mkhomedir.*/session\trequired\tpam_mkhomedir.so umask=0022/g;' /etc/pam.d/common-session
sed
수년간 사용 했지만 대부분 s
명령 만 사용 했습니다 . hold
및 pattern
공간 스왑은 내 능력 밖입니다.
보호 된 파일에 쓰는 경우 @drAlberT 및 @ rubo77의 답변은 sudo 할 수 없으므로 작동하지 않을 수 있습니다 >>
. 유사 간단한 해결책은 , 다음, 사용하는 것입니다 tee --append
(맥 OS에, 또는 tee -a
) :
LINE='include "/configs/projectname.conf"'
FILE=lighttpd.conf
grep -qF "$LINE" "$FILE" || echo "$LINE" | sudo tee --append "$FILE"
언젠가 다른 사람 이이 코드를 "레거시 코드"로 처리 해야하는 경우 다음과 같이 덜 이질적인 코드를 작성하면 그 사람에게 감사하게 될 것입니다.
grep -q -F 'include "/configs/projectname.conf"' lighttpd.conf
if [ $? -ne 0 ]; then
echo 'include "/configs/projectname.conf"' >> lighttpd.conf
fi
또 다른 sed 솔루션은 항상 마지막 줄에 추가하고 기존의 것을 삭제하는 것입니다.
sed -e '$a\' -e '<your-entry>' -e "/<your-entry-properly-escaped>/d"
"적절하게 이스케이프 처리됨"은 항목과 일치하는 정규식을 배치하는 것, 즉 실제 항목에서 모든 정규식 컨트롤을 이스케이프 처리하는 것, 즉 ^ $ / *? + () 앞에 백 슬래시를 두는 것을 의미합니다.
이것은 파일의 마지막 줄에서 실패하거나 줄 바꿈이 없으면 확실하지 않지만 일부 멋진 분기로 처리 할 수 있습니다.
sed -i.bak -e '$a\' -e "$NEW_IP\t\t$HOST.domain\t$HOST" -e "/.*$HOST/d" /etc/hosts
sed -i -e '/<entry>/d; $a <entry>'
d
명령이 sed 프로그램을 다시 시작 a
하여 삭제가 마지막 행에있는 경우 ppend 를 건너 뛰기 때문에 이것이 실패한다고 생각합니다 .
a
보류하기 전에 실행해야 d
합니다 : sed -i -e '$a<entry>' -e '/<entry>/d'
. 원래 답변과 거의 동일합니다.
grep을 사용한 답변이 잘못되었습니다. 전체 행을 일치 시키려면 -x 옵션을 추가해야합니다. 그렇지 않으면 #text to add
정확하게 추가 할 때 와 같은 행 이 여전히 일치합니다 text to add
.
올바른 해결책은 다음과 같습니다.
grep -qxF 'include "/configs/projectname.conf"' foo.bar || echo 'include "/configs/projectname.conf"' >> foo.bar
sed 사용 : 줄 끝에 삽입합니다. 물론 평소대로 변수를 전달할 수도 있습니다.
grep -qxF "port=9033" $light.conf
if [ $? -ne 0 ]; then
sed -i "$ a port=9033" $light.conf
else
echo "port=9033 already added"
fi
oneliner sed 사용
grep -qxF "port=9033" $lightconf || sed -i "$ a port=9033" $lightconf
echo를 사용하면 루트에서는 작동하지 않지만 다음과 같이 작동합니다. 그러나 암호를 요구할 수 있기 때문에 원하는 작업을 자동화하지 못합니다.
특정 사용자의 루트에서 편집하려고 할 때 문제가 발생했습니다. $username
이전을 추가하는 것이 나를 위해 수정되었습니다.
grep -qxF "port=9033" light.conf
if [ $? -ne 0 ]; then
sudo -u $user_name echo "port=9033" >> light.conf
else
echo "already there"
fi
필요한 쓰기 권한이 제한된 파일을 편집해야했습니다 sudo
. ghostdog74의 답변에서 작업하고 임시 파일을 사용하십시오.
awk 'FNR==NR && /configs.*projectname\.conf/{f=1;next}f==0;END{ if(!f) { print "your line"}} ' file > /tmp/file
sudo mv /tmp/file file
crudini
(그러나 lighthttpd 아직에 대한) 좋은 옵션이 될 수있다