Cara Menggunakan set dan pipefail di Bash Script di Linux
Diterbitkan: 2022-06-25 Perintah set
dan pipefail
Linux menentukan apa yang terjadi ketika terjadi kegagalan dalam skrip Bash. Ada lebih banyak untuk dipikirkan daripada harus berhenti atau harus terus berlanjut.
TERKAIT: Panduan Pemula untuk Shell Scripting: Dasar-dasar
Skrip Bash dan Kondisi Kesalahan
Skrip bash shell sangat bagus. Mereka cepat menulis dan tidak perlu dikompilasi. Setiap tindakan berulang atau multi-tahap yang perlu Anda lakukan dapat dibungkus dengan skrip yang nyaman. Dan karena skrip dapat memanggil salah satu utilitas Linux standar, Anda tidak terbatas pada kemampuan bahasa shell itu sendiri.
Tetapi masalah dapat muncul ketika Anda memanggil utilitas atau program eksternal. Jika gagal, utilitas eksternal akan menutup dan mengirim kode kembali ke shell, dan bahkan mungkin mencetak pesan kesalahan ke terminal. Tetapi skrip Anda akan terus diproses. Mungkin bukan itu yang Anda inginkan. Jika kesalahan terjadi di awal eksekusi skrip, hal itu dapat menyebabkan masalah yang lebih buruk jika skrip lainnya diizinkan untuk dijalankan.
Anda dapat memeriksa kode pengembalian dari setiap proses eksternal setelah selesai, tetapi itu menjadi sulit ketika proses disalurkan ke proses lain. Kode pengembalian akan berasal dari proses di ujung pipa, bukan yang di tengah yang gagal. Tentu saja, kesalahan juga dapat terjadi di dalam skrip Anda, seperti mencoba mengakses variabel yang tidak diinisialisasi.
Perintah set
dan pipefile
memungkinkan Anda memutuskan apa yang terjadi ketika kesalahan seperti ini terjadi. Mereka juga memungkinkan Anda mendeteksi kesalahan bahkan ketika itu terjadi di tengah-tengah rantai pipa.
Berikut cara menggunakannya.
Mendemonstrasikan Masalah
Inilah skrip Bash yang sepele. Ini menggemakan dua baris teks ke terminal. Anda dapat menjalankan skrip ini jika Anda menyalin teks ke editor dan menyimpannya sebagai "script-1.sh."
#!/bin/bash echo Ini akan terjadi dulu echo Ini akan terjadi kedua
Untuk membuatnya dapat dieksekusi, Anda harus menggunakan chmod
:
chmod +x skrip-1.sh
Anda harus menjalankan perintah itu pada setiap skrip jika Anda ingin menjalankannya di komputer Anda. Mari kita jalankan skripnya:
./script-1.sh
Dua baris teks dikirim ke jendela terminal seperti yang diharapkan.
Mari kita ubah skripnya sedikit. Kami akan meminta ls
untuk mencantumkan detail file yang tidak ada. Ini akan gagal. Kami menyimpan ini sebagai "script-2.sh" dan membuatnya dapat dieksekusi.
#!/bin/bash echo Ini akan terjadi dulu ls nama file imajiner echo Ini akan terjadi kedua
Ketika kami menjalankan skrip ini, kami melihat pesan kesalahan dari ls
.
./script-2.sh
Meskipun perintah ls
gagal, skrip terus berjalan. Dan meskipun ada kesalahan selama eksekusi skrip, kode pengembalian dari skrip ke shell adalah nol, yang menunjukkan keberhasilan. Kita dapat memeriksa ini menggunakan echo dan $?
variabel yang menyimpan kode pengembalian terakhir yang dikirim ke shell.
gema $?
Nol yang dilaporkan adalah kode pengembalian dari gema kedua dalam skrip. Jadi ada dua masalah dengan skenario ini. Yang pertama adalah skrip mengalami kesalahan tetapi tetap berjalan. Itu dapat menyebabkan masalah lain jika sisa skrip mengharapkan atau bergantung pada tindakan yang gagal benar-benar berhasil. Dan yang kedua adalah jika skrip atau proses lain perlu memeriksa keberhasilan atau kegagalan skrip ini, itu akan mendapatkan pembacaan yang salah.
Opsi set -e
Opsi set -e
(exit) menyebabkan skrip keluar jika salah satu proses yang dipanggil menghasilkan kode pengembalian bukan nol. Segala sesuatu yang bukan nol dianggap sebagai kegagalan.
Dengan menambahkan opsi set -e
ke awal skrip, kita dapat mengubah perilakunya. Ini adalah "script-3.sh."
#!/bin/bash set -e echo Ini akan terjadi dulu ls nama file imajiner echo Ini akan terjadi kedua
Jika kita menjalankan skrip ini, kita akan melihat efek dari set -e
.
./script-3.sh
gema $?
Skrip dihentikan dan kode pengembalian yang dikirim ke shell adalah nilai bukan nol.
Menangani Kegagalan di Pipa
Pemipaan menambah kompleksitas masalah. Kode pengembalian yang keluar dari urutan perintah yang disalurkan adalah kode pengembalian dari perintah terakhir dalam rantai. Jika ada kegagalan dengan perintah di tengah rantai, kami kembali ke titik awal. Kode pengembalian itu hilang, dan skrip akan terus diproses.
Kita dapat melihat efek dari perintah pemipaan dengan kode pengembalian yang berbeda menggunakan built-in shell true
dan false
. Kedua perintah ini tidak lebih dari menghasilkan kode pengembalian masing-masing nol atau satu.
BENAR
gema $?
Salah
gema $?
Jika kita mem-pipe false
ke true
—dengan false
mewakili proses yang gagal—kita mendapatkan kode pengembalian true
dari nol.
salah | BENAR
gema $?
Bash memang memiliki variabel array yang disebut PIPESTATUS
, dan ini menangkap semua kode yang dikembalikan dari setiap program dalam rantai pipa.
salah | benar | salah | BENAR
echo "${PIPESTATUS[0]} ${PIPESTATUS[1]} ${PIPESTATUS[2]} ${PIPESTATUS[3]}"
PIPESTATUS
hanya menyimpan kode pengembalian sampai program berikutnya berjalan, dan mencoba menentukan kode pengembalian mana yang sesuai dengan program mana yang bisa menjadi sangat berantakan dengan sangat cepat.
Di sinilah set -o
(opsi) dan pipefail
masuk. Ini adalah "script-4.sh." Ini akan mencoba menyalurkan konten file yang tidak ada ke wc
.
#!/bin/bash set -e echo Ini akan terjadi dulu kucing script-99.sh | wc -l echo Ini akan terjadi kedua
Ini gagal, seperti yang kami harapkan.
./script-4.sh
gema $?
Nol pertama adalah output dari wc
, memberi tahu kami bahwa itu tidak membaca baris apa pun untuk file yang hilang. Nol kedua adalah kode kembali dari perintah echo
kedua.
Kami akan menambahkan -o pipefail
, simpan sebagai "script-5.sh", dan membuatnya dapat dieksekusi.
#!/bin/bash set -eo pipefail echo Ini akan terjadi dulu kucing script-99.sh | wc -l echo Ini akan terjadi kedua
Mari kita jalankan itu dan periksa kode pengembaliannya.
./script-5.sh
gema $?
Skrip berhenti dan perintah echo
kedua tidak dijalankan. Kode pengembalian yang dikirim ke shell adalah satu, menunjukkan kegagalan dengan benar.
TERKAIT: Cara Menggunakan Perintah Echo di Linux
Menangkap Variabel yang Tidak Diinisialisasi
Variabel yang tidak diinisialisasi bisa sulit dikenali dalam skrip dunia nyata. Jika kita mencoba untuk echo
nilai dari variabel yang tidak diinisialisasi, echo
hanya mencetak baris kosong. Itu tidak memunculkan pesan kesalahan. Sisa skrip akan terus dieksekusi.
Ini adalah skrip-6.sh.
#!/bin/bash set -eo pipefail echo "$notset" echo "Perintah gema lainnya"
Kami akan menjalankannya dan mengamati perilakunya.
./script-6.sh
gema $?
Script melangkah di atas variabel yang tidak diinisialisasi, dan terus dieksekusi. Kode pengembalian adalah nol. Mencoba menemukan kesalahan seperti ini dalam skrip yang sangat panjang dan rumit bisa sangat sulit.
Kita dapat menjebak jenis kesalahan ini menggunakan opsi set -u
(tidak disetel). Kami akan menambahkannya ke koleksi opsi set kami yang berkembang di bagian atas skrip, simpan sebagai "script-7.sh", dan membuatnya dapat dieksekusi.
#!/bin/bash set -eou pipefail echo "$notset" echo "Perintah gema lainnya"
Mari kita jalankan skripnya:
./script-7.sh
gema $?
Variabel yang tidak diinisialisasi terdeteksi, skrip berhenti, dan kode kembali disetel ke satu.
Opsi -u
(tidak disetel) cukup cerdas untuk tidak dipicu oleh situasi di mana Anda dapat secara sah berinteraksi dengan variabel yang tidak diinisialisasi.
Dalam "script-8.sh", skrip memeriksa apakah variabel New_Var
diinisialisasi atau tidak. Anda tidak ingin skrip berhenti di sini, dalam skrip dunia nyata Anda akan melakukan pemrosesan lebih lanjut dan menangani sendiri situasinya.
Perhatikan bahwa kami telah menambahkan opsi -u
sebagai opsi kedua dalam pernyataan set. Opsi -o pipefail
harus berada di urutan terakhir.
#!/bin/bash set -euo pipefail if [ -z "${Var_Baru:-}" ]; kemudian echo "New_Var tidak memiliki nilai yang ditetapkan untuk itu." fi
Dalam "script-9.sh", variabel yang tidak diinisialisasi diuji dan jika tidak diinisialisasi, nilai default disediakan sebagai gantinya.
#!/bin/bash set -euo pipefail nilai_default=484 Nilai=${Var_Baru:-$nilai_default} echo "Var_Baru=$Nilai"
Skrip diizinkan untuk dijalankan hingga selesai.
./script-8.sh
./script-9.sh
Disegel dengan kapak
Opsi praktis lainnya untuk digunakan adalah opsi set -x
(eksekusi dan cetak). Saat Anda menulis skrip, ini bisa menjadi penyelamat. itu mencetak perintah dan parameternya saat dieksekusi.
Ini memberi Anda bentuk jejak eksekusi "kasar dan siap" yang cepat. Mengisolasi kelemahan logika dan menemukan bug menjadi jauh lebih mudah.
Kami akan menambahkan opsi set -x ke "script-8.sh", simpan sebagai "script-10.sh", dan membuatnya dapat dieksekusi.
#!/bin/bash set -euxo pipefail if [ -z "${Var_Baru:-}" ]; kemudian echo "New_Var tidak memiliki nilai yang ditetapkan untuk itu." fi
Jalankan untuk melihat garis jejak.
./script-10.sh
Menemukan bug dalam skrip contoh sepele ini mudah. Ketika Anda mulai menulis skrip yang lebih terlibat, opsi ini akan membuktikan nilainya.