Cara Menggunakan set dan pipefail di Bash Script di Linux

Diterbitkan: 2022-06-25
Terminal Linux pada layar laptop dengan latar belakang biru.
fatmawati achmad zaenuri/Shutterstock.com

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.

Apa itu Bash Shell, dan Mengapa Sangat Penting untuk Linux?
TERKAIT Apa itu Bash Shell, dan Mengapa Sangat Penting untuk Linux?

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 

Menjalankan skrip sederhana tanpa kesalahan.

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 

Menjalankan skrip dan menghasilkan kondisi gagal.

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 $? 

Memeriksa kode pengembalian untuk skrip terakhir yang dieksekusi.

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 $? 

Mengakhiri skrip pada kondisi kesalahan, dan mengatur kode pengembalian dengan benar.

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 $? 

Perintah bawaan bash shell true dan false.

Jika kita mem-pipe false ke true —dengan false mewakili proses yang gagal—kita mendapatkan kode pengembalian true dari nol.

 salah | BENAR
 gema $? 

Piping salah menjadi benar.

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]}" 

Menggunakan PIPESTATUS untuk melihat kode pengembalian semua program dalam rantai pipa.

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 $? 

Menjalankan skrip dengan kesalahan dalam rantai pipa.

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 $? 

Menjalankan skrip yang menjebak kesalahan dalam rantai pipa dan menyetel kode pengembalian dengan benar.

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 $? 

Menjalankan skrip yang tidak menangkap variabel yang tidak diinisialisasi.

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.

Cara Bekerja dengan Variabel di Bash
TERKAIT Cara Bekerja dengan Variabel di Bash

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 $? 

Menjalankan skrip yang menangkap variabel yang tidak diinisialisasi.

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 

Menjalankan dua skrip di mana variabel yang tidak diinisialisasi ditangani secara internal, dan opsi -u tidak terpicu.

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 

Menjalankan skrip dengan -x trace lines yang ditulis ke terminal.

Menemukan bug dalam skrip contoh sepele ini mudah. Ketika Anda mulai menulis skrip yang lebih terlibat, opsi ini akan membuktikan nilainya.