Linux awk confrontando due file CSV e creando un nuovo file con un flag
-
15-11-2019 - |
Domanda
Ho 2 file CSV che devo confrontare e ottenere la differenza per un file appena formattato. I campioni sono riportati di seguito.
Vecchio file
.
DTL,11111111,1111111111111111,11111111111,Y,N,xx,xx
DTL,22222222,2222222222222222,22222222222,Y,Y,cc,cc
DTL,33333333,3333333333333333,33333333333,Y,Y,dd,dd
DTL,44444444,4444444444444444,44444444444,Y,Y,ss,ss
DTL,55555555,5555555555555555,55555555555,Y,Y,qq,qq
Nuovo file
.
DTL,11111111,1111111111111111,11111111111,Y,Y,xx,xx
DTL,22222222,2222222222222222,22222222222,Y,N,cc,cc
DTL,44444444,4444444444444444,44444444444,Y,Y,ss,ss
DTL,55555555,5555555555555555,55555555555,Y,Y,qq,qq
DTL,77777777,7777777777777777,77777777777,N,N,ee,ee
Voglio confrontare i vecchi e nuovi file CSV e per trovare le modifiche che si sono effettuate nel nuovo file e aggiornano un flag per indicare queste modifiche
U - Se il nuovo record di file è aggiornato D - Se un disco esistente nel vecchio file viene cancellato nel nuovo file N - Se un disco esistente nel nuovo file non è disponibile nel vecchio file
Il file di output di esempio è questo.
.
DTL,11111111,1111111111111111,11111111111,Y,Y,xx,xx U
DTL,22222222,2222222222222222,22222222222,Y,N,cc,cc U
DTL,33333333,3333333333333333,33333333333,Y,Y,dd,dd D
DTL,77777777,7777777777777777,77777777777,N,N,ee,ee N
Ho usato il comando diff ma ripeterà anche il record aggiornato che non voglio.
.
DTL,11111111,1111111111111111,11111111111,Y,N,xx,xx
DTL,22222222,2222222222222222,22222222222,Y,Y,cc,cc
DTL,33333333,3333333333333333,33333333333,Y,Y,dd,dd
---
DTL,11111111,1111111111111111,11111111111,Y,Y,xx,xx
DTL,22222222,2222222222222222,22222222222,Y,N,cc,cc
5a5
DTL,77777777,7777777777777777,77777777777,N,N,ee,ee
Ho usato un comando awk singolo riga per filtrare i miei record pure? / p>
.
awk 'NR==FNR{A[$1];next}!($1 in A)' FS=: old.csv new.csv
Il problema con questo è non è prendermi i record appartenenti solo al vecchio file.
che è
.
DTL,33333333,3333333333333333,33333333333,Y,Y,dd,dd
Ho avviato anche uno script bash guidato per l'ahieve, ma non ha trovato molto aiuto con un buon esempio.
. myscript.awk
BEGIN {
FS = "," # input field seperator
OFS = "," # output field seperator
}
NR > 1 {
#flag
# N - new record D- Deleted U - Updated
id = $1
name = $2
flag = 'N'
# This prints the columns in the new order. The commas tell Awk to use the character set in OFS
print id,name,flag
}
>> awk -f myscript.awk old.csv new.csv > formatted.csv
Soluzione
This might work for you:
diff -W999 --side-by-side OLD NEW |
sed '/^[^\t]*\t\s*|\t\(.*\)/{s//\1 U/;b};/^\([^\t]*\)\t*\s*<$/{s//\1 D/;b};/^.*>\t\(.*\)/{s//\1 N/;b};d'
DTL,11111111,1111111111111111,11111111111,Y,Y,xx,xx U
DTL,22222222,2222222222222222,22222222222,Y,N,cc,cc U
DTL,33333333,3333333333333333,33333333333,Y,Y,dd,dd D
DTL,77777777,7777777777777777,77777777777,N,N,ee,ee N
an awk solution along the same lines:
diff -W999 --side-by-side OLD NEW |
awk '/[|][\t]/{split($0,a,"[|][\t]");print a[2]" U"};/[\t] *<$/{split($0,a,"[\t]* *<$");print a[1]" D"};/>[\t]/{split($0,a,">[\t]");print a[2]" N"}'
DTL,11111111,1111111111111111,11111111111,Y,Y,xx,xx U
DTL,22222222,2222222222222222,22222222222,Y,N,cc,cc U
DTL,33333333,3333333333333333,33333333333,Y,Y,dd,dd D
DTL,77777777,7777777777777777,77777777777,N,N,ee,ee N
Altri suggerimenti
A good starting point would probably be:
diff -e OLD NEW
This outputs:
5a
DTL,77777777,7777777777777777,77777777777,N,N,ee,ee
.
1,3c
DTL,11111111,1111111111111111,11111111111,Y,Y,xx,xx
DTL,22222222,2222222222222222,22222222222,Y,N,cc,cc
Meaning that it Added a record on line 5 (5a) and changed the records on lines 1 and 3 (1,3c).
If you can't use this format as-is (which would be good to use a standard) then you would need to write a script which converts it to the format that you describe.