ฉันจะป้องกันไม่ให้ sed -i ทำลาย symlink ได้อย่างไร


21

ทำไมsed -iการดำเนินการของ symlink จึงทำลายลิงก์นั้นและแทนที่ด้วยไฟล์ปลายทาง จะหลีกเลี่ยงสิ่งนี้ได้อย่างไร

เช่น.

$ ls -l pet*
-rw-rw-r-- 1 madneon madneon 4 mar 23 16:46 pet
lrwxrwxrwx 1 madneon madneon 6 mar 23 16:48 pet_link -> pet

$ sed -i 's/cat/dog/' pet_link

$ ls -l pet*
-rw-rw-r-- 1 madneon madneon 4 mar 23 16:48 pet
-rw-rw-r-- 1 madneon madneon 4 mar 23 16:49 pet_link

และทำไมมันไม่ถือว่าเป็นข้อผิดพลาด?

คำตอบ:


25

กระบวนการ-i/ --in-placeflag แก้ไขไฟล์ในสถานที่ ตามค่าเริ่มต้นให้sedอ่านไฟล์ที่กำหนดประมวลผลไฟล์ที่ส่งออกเป็นไฟล์ชั่วคราวจากนั้นคัดลอกไฟล์ชั่วคราวไปที่ต้นฉบับโดยไม่ตรวจสอบว่าไฟล์ต้นฉบับนั้นเป็น symlink หรือไม่

GNU sedมีการ--follow-symlinksตั้งค่าสถานะซึ่งทำให้มันทำงานตามที่คุณต้องการ:

$ echo "cat" > pet
$ ln --symbolic pet pet_link
$ sed --in-place --follow-symlinks 's/cat/dog/' pet_link
$ cat pet
dog

6
มันไม่ได้แก้ไขไฟล์ในสถานที่ แต่แก้ไข temp copy ของไฟล์ในไดเรกทอรีปัจจุบันจากนั้นย้าย temp copy ไปที่ต้นฉบับ
mikeserv

@mikeserv ฉันข้ามรายละเอียดการใช้งานไปเพราะคำถามเกี่ยวกับส่วนต่อประสาน ดีที่จะรู้ว่าขอบคุณ!
Anko

1

ไม่ใช่ข้อผิดพลาดนี่คือการออกแบบเนื่องจากsedเป็นS tream ED itor ไม่ใช่ตัวแก้ไขไฟล์ โดยทั่วไปจะทำการคัดลอกและแทนที่ไฟล์ต้นฉบับด้วยการคัดลอก BashFAQ

หรือคุณสามารถใช้exคำสั่งแทนซึ่งมีไวยากรณ์ที่คล้ายคลึงกันสำหรับการทดแทนเช่น

ex +%s/cat/dog/ge -scwq pet_link

หรือหลายไฟล์:

ex "+bufdo! %s/cat/dog/ge" -scxa **/pet_link*

มันจะไม่ทำลายลิงก์สัญลักษณ์

ที่เกี่ยวข้อง: ฉันจะป้องกันไม่ให้ sed ถูกทำลายจาก hardinks ได้อย่างไร


0

ฉันพบว่าวิธีนี้ใช้งานได้ดี (การรักษาลิงก์และสัญลักษณ์ทั้งสอง):

sed 's/cat/dog/' pet_link > pet_link.tmp
cat pet_link.tmp > pet_link
rm pet_link.tmp

0

มีวิธีแก้ไขที่บางครั้งเราใช้เพื่อเขียนไปยังไฟล์เดียวกับที่อ่านจาก นี่คือข้อความที่ตัดตอนมาจากหน้าคน:

   sponge reads standard input and writes it out to the specified file.
   Unlike a shell redirect, sponge soaks up all its input before opening
   the output file. This allows constructing pipelines that read from and
   write to the same file.

   It also creates the output file atomically by renaming a temp file into
   place, and preserves the permissions of the output file if it already
   exists. If the output file is a special file or symlink, the data will
   be written to it.

นี่เป็นตัวอย่างที่แสดงให้เห็นว่ามันสามารถรักษาลิงก์สัญลักษณ์แม้ว่าฉันมักจะใช้มันเพื่อรักษา inodes:

# Utility functions: print-as-echo, print-line-with-visual-space.
pe() { for _i;do printf "%s" "$_i";done; printf "\n"; }
pl() { pe;pe "-----" ;pe "$*"; }

rm -f pet pet_link
echo "cat" > pet
pl " Input data file $FILE:"
head -v pet

pl " Results, before sed:"
ln --symbolic pet pet_link
ls -ligG pet pet_link
# sed --in-place --follow-symlinks 's/cat/dog/' pet_link
pe
pe " Results, after sed:"
sed 's/cat/dog/' pet_link | sponge pet_link
head -v pet
ls -ligG pet pet_link

ซึ่งผลิต:

-----
 Input data file data1:
==> pet <==
cat

-----
 Results, before sed:
1571283 -rw-r--r-- 1 4 Nov 26 23:03 pet
1571286 lrwxrwxrwx 1 3 Nov 26 23:03 pet_link -> pet

 Results, after sed:
==> pet <==
cat
1571283 -rw-r--r-- 1 4 Nov 26 23:03 pet
1571286 lrwxrwxrwx 1 3 Nov 26 23:03 pet_link -> pet

ในระบบเช่น:

OS, ker|rel, machine: Linux, 3.16.0-4-amd64, x86_64
Distribution        : Debian 8.9 (jessie) 
bash GNU bash 4.3.30

รหัสฟองน้ำมีอยู่ในแพ็คเกจmoreutils - รายละเอียดบางอย่าง:

sponge  soak up standard input and write to a file (man)
Path    : /usr/bin/sponge
Package : moreutils
Home    : http://kitenet.net/~joey/code/moreutils/
Version : 0.52
Type    : ELF 64-bit LSB executable, x86-64, version 1 (SYS ...)

ที่ร้านของเราเราเขียนเวอร์ชันที่เขียนไปยังไฟล์ชั่วคราวสำหรับกรณีของไฟล์ที่มีขนาดใหญ่มาก

แพ็คเกจมีให้บริการใน Debian, Fedora, macOS (ผ่านทาง brew) ฯลฯ ... ไชโย

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.