¿Cómo cambiar un archivo en el lugar usando awk? (como con "sed -i")

11

Tengo un awkguión new.awk:

 BEGIN { FS=OFS="," }
 NR==1 {
for (i=1; i<=NF; i++) {
    f[$i] = i
}
   } 
  NR > 1 {
 begSecs= mktime(gensub(/[":-]/," ","g",$(f["DateTime"])))
 endSecs = begSecs + $(f["TotalDuration"])
 $(f["CallEndTime"]) = strftime("%Y-%m-%d %H:%M:%S", endSecs)
 }
 { print }

Estoy llamando a esto en shell

awk new.awk sample.csv

... pero puedo ver los cambios en la terminal. ¿Cómo hacer el cambio en el lugar en el archivo, como cuando se usa sed -i?

mittu
fuente

Respuestas:

16

GNU awk(que se encuentra comúnmente en sistemas Linux), desde la versión 4.1.0, puede incluir una " awkbiblioteca fuente" con -io --includeen la línea de comando. Una de las bibliotecas de origen que se distribuye con GNU awkes una llamada inplace:

$ cat file
hello
there
$ awk -i inplace '/hello/ { print "oh,", $0 }' file
$ cat file
oh, hello

Como puede ver, esto hace que la salida del awkcódigo reemplace el archivo de entrada. La frase que dice thereno se guarda ya que el programa no la genera.

Con un awkscript en un archivo, lo usarías como

awk -i inplace -f script.awk datafile

Si la awkvariable INPLACE_SUFFIXse establece en una cadena, entonces la biblioteca haría una copia de seguridad del archivo original con eso como sufijo de nombre de archivo.

awk -i inplace -v INPLACE_SUFFIX=.bak -f script.awk datafile

Si tiene varios archivos de entrada, cada archivo se editará individualmente en el lugar. Pero puede desactivar la edición en el lugar para un archivo (o un conjunto de archivos) mediante el uso inplace=0de la línea de comando antes de ese archivo:

awk -i inplace -f script.awk file1 file2 inplace=0 file3 inplace=1 file4

En el comando anterior, file3no se editaría en el lugar.


Para una "edición local" más portátil de un solo archivo, use

tmpfile=$(mktemp)
cp file "$tmpfile" &&
awk '...some program here...' "$tmpfile" >file
rm "$tmpfile"

Esto copiaría el archivo de entrada a una ubicación temporal, luego aplicaría el awkcódigo en el archivo temporal mientras redirige al nombre de archivo original.

Hacer las operaciones en este orden (ejecutándose awken el archivo temporal, no en el archivo original) asegura que los metadatos del archivo (permisos y propiedad) del archivo original no se modifiquen.

Kusalananda
fuente
4

Prueba esto.

awk  new.awk sample.csv > tmp.csv && mv -f tmp.csv sample.csv
  • redirigir la salida a un archivo temporal.
  • luego mueva el contenido del archivo temporal al archivo original.
Siva
fuente
1
Si observa el código fuente in situ de awk , esto es exactamente lo que hace: crea un archivo temporal, redirige stdout allí y al final lo renombra al archivo de entrada.
chx