цикл awk для каждой строки в файле

цикл awk для каждой строки в файле

Мне нужно выяснить, как выполнить цикл по текстовому файлу и для каждой строки взять поля имени, номера проекта и адреса электронной почты, а затем заменить их в шаблоне электронного письма для отправки.

Итак, это текстовый файл с именем send.txt:

Project 1,Jack,Chen,06,12,[email protected]
Project 2,Emily,Weiss,06,12,[email protected]
Project 3,Mary,Gonzalas,06,12,[email protected]

а это шаблон электронного письма под названием Reminder.email:

Dear __FULLNAME__: 

This is a kindly reminder that our __Project__ meeting will be held on today. 

Best Regards, 
CIS5027

Итак, для каждой строки текстового файла мне нужно заменить в этом шаблоне электронного письма поляПОЛНОЕ ИМЯ: , иПроект. С соответствующими значениями, которые я могу сделать для первой строки, однако я не могу сделать это для каждой строки.

Это мой сценарий.

#

!/bin/sh
#Start your code from here

date= date "+%m/%d/%y"
print $date

#The following line is to scan a file called Event-member.data and return any lines with todays date and save them to a file called sendtoday.txt

grep -n $(date +"%m,%d") Event-member.data > sendtoday.txt

#The following line is to remove the first to characters of the file created above sendtoday.txt and output that to a file called send.txt.

cat sendtoday.txt | sed 's/^..//' > send.txt


#This is where im having trouble. When storing the values for the variables below of name, email, project #. The value of NR==1 thus it never goes through the rest of the lines. I've tried different solutions but none seem to work.

p=$(awk -F ',' 'NR==1{print $1}' send.txt)
n=$(awk -F ',' 'NR==1{print $2}' send.txt)
l=$(awk -F ',' 'NR==1{print $3}' send.txt)
e=$(awk -F ',' 'NR==1{print $6}' send.txt)

echo $p  $n  $l  $e

#This part is to replace the values in the email template using sed and save the modified template as sendnow.txt.  

sed -e "s/__FULLNAME__:/\ $n $l :/g;s/__Project__/\ $p /g" Reminder.email > sendnow.txt

cat sendnow.txt


#Yet to be written ... send out modified email templates.
exit 0

########

Вот что он выдает:

06/12/14
Project 1 Jack Chen [email protected]
Dear  Jack Chen : 

This is a kindly reminder that our  Project 1  meeting will be held on today. 

Best Regards, 
CIS5027

Как видите, поля были заменены правильно, но только для Джека Чена. В файле send.txt было 3 строки, поэтому должно быть 3 измененных версии шаблона выше.

решение1

Нет смысла использовать awkдля этого. Вы можете сделать это прямо в оболочке, используяread. Общий формат — это read foo barсохранение первого поля как $fooи остальной части каждой строки как $bar. Так что в вашем случае вы бы сделали что-то вроде:

while IFS="," read p n l foo bar e; do 
    sed -e "s/__FULLNAME__:/\ $n $l :/g;s/__Project__/\ $p /g" Reminder.email; 
done < file

Это IFSразделитель полей ввода, который при установке на ,считывает поля, разделенные запятыми. Это позволяет вам брать каждое поле и сохранять его в переменной. Обратите внимание, что я использовал две дополнительные переменные fooи bar. Это потому, что каждому полю нужно свое имя переменной, а у вас есть 6 полей. Если вы дадите только 4 имени переменных, 4-е ( $e) будет содержать поля с 4-го по последнее.


Теперь, в вашем скрипте есть различные другие синтаксические ошибки. Во-первых, строка shebang неверна, вам нужно #! /bin/sh, не может быть пустой строки между #!и /bin/sh. Также, чтобы назначитьвыходкоманды к переменной, вам нужно использовать var=`command`or, желательно var=$(command)формат. В противном случае переменной присваивается сама команда как строка, а не ее вывод. Наконец,printне то, что вы думаете. Вы ищете printfили echo. Поэтому лучшим способом написать свой сценарий будет:

#!/bin/sh

date=$(date "+%m/%d/%y")
echo $date

## The following line is to scan a file called Event-member.data 
## and return any lines with todays date and save them to a file 
## called sendtoday.txt
grep -n $(date +"%m,%d") Event-member.data > sendtoday.txt

## The following line is to remove the first to characters of the file
## created above sendtoday.txt and output that to a file called
## send.txt. 
## I rewrote this to avoid the useless use of cat.
sed 's/^..//' sendtoday.txt > send.txt


## This is where you use read
while IFS="," read p n l foo bar e; do 
    sed -e "s/__FULLNAME__:/\ $n $l :/g;s/__Project__/\ $p /g" Reminder.email > sendnow.txt
    cat sendnow.txt
    ## This is where you need to add the code that sends the emails. Something
    ## like this:
    sendmail $e < sendnow.txt

done < send.txt

exit 0

########

решение2

Вы использовали условие NR==1 NR==1{print $1}. Это означает, что будет рассмотрена первая строка send.txt. Используйте условие NR==2, чтобы получить вторую строку и так далее. ИЛИ используйте цикл, чтобы пройти по всем строкам, например,

while read line
do
  p=`echo $line | awk -F '.' '{print $1}'`
  n=`echo $line | awk -F '.' '{print $2}'`
  l=`echo $line | awk -F '.' '{print $3}'`
  e=`echo $line | awk -F '.' '{print $1}'`

  sed -e "s/\__FULLNAME\__:/\ $n $l :/g;s/\__Project__/\ $p /g" Reminder.email > sendnow.txt

done<send.txt

решение3

может быть, оберните ваш скрипт во что-то вроде

for i in $(cat send.txt); do echo "line: $i"; done

?

решение4

cat xtmp |  awk '{s++; do_aything_you_want_here }'

**s++ пройтись по каждой строке **

cat xtmp
111
222
333
444

cat xtmp |  awk '{s++; print($1,"QQQ") }'
111 QQQ
222 QQQ
333 QQQ
444 QQQ

cat xtmp |  awk '{s++; print($1+3,"QQQ") }'
114 QQQ
225 QQQ
336 QQQ
447 QQQ

cat xtmp |  awk '{s++; for(i=1; i<=4 ; i++) print $1 }'  # this will repeat each line 4 time

cat tmp | awk '{s++; a=a+$1; print(s" "$1" "a)}' instead of a=a+$1 you can use a+=$1
1 111 111
2 222 333
3 333 666
4 444 1110

Связанный контент