Cara Menggunakan getopts untuk Mengurai Opsi Skrip Shell Linux

Diterbitkan: 2022-06-25
Layar laptop menampilkan teks terminal.
fatmawati achmad zaenuri/Shutterstock.com

Apakah Anda ingin skrip shell Linux Anda menangani opsi dan argumen baris perintah dengan lebih anggun? Bash getopts bawaan memungkinkan Anda mengurai opsi baris perintah dengan kemahiran—dan itu juga mudah. Kami tunjukkan caranya.

Memperkenalkan getopts bawaan

Melewati nilai ke dalam skrip Bash adalah masalah yang cukup sederhana. Anda memanggil skrip Anda dari baris perintah atau dari skrip lain dan memberikan daftar nilai Anda di belakang nama skrip. Nilai-nilai ini dapat diakses di dalam skrip Anda sebagai variabel, dimulai dengan $1 untuk variabel pertama, $2 untuk variabel kedua, dan seterusnya.

Tetapi jika Anda ingin meneruskan opsi ke skrip, situasinya dengan cepat menjadi lebih kompleks. Ketika kami mengatakan opsi, yang kami maksud adalah opsi, flag, atau sakelar yang dapat ditangani oleh program seperti ls . Mereka didahului oleh tanda hubung “ - ” dan biasanya bertindak sebagai indikator program untuk mengaktifkan atau menonaktifkan beberapa aspek fungsinya.

Cara Menggunakan Perintah ls untuk Mendaftar File dan Direktori di Linux
TERKAIT Cara Menggunakan Perintah ls untuk Membuat Daftar File dan Direktori di Linux

Perintah ls memiliki lebih dari 50 opsi, terutama terkait dengan memformat outputnya. Opsi -X (urutkan berdasarkan ekstensi) mengurutkan output menurut abjad berdasarkan ekstensi file. Opsi -U (tidak disortir) daftar berdasarkan urutan direktori.

Pilihannya hanya itu—mereka opsional. Anda tidak tahu opsi mana—jika ada—pengguna akan memilih untuk digunakan, dan Anda juga tidak tahu dalam urutan mana mereka dapat mencantumkannya di baris perintah. Ini meningkatkan kerumitan kode yang diperlukan untuk mengurai opsi.

Hal-hal menjadi lebih rumit masih jika beberapa opsi Anda mengambil argumen, yang dikenal sebagai argumen opsi , Misalnya, opsi ls -w (lebar) mengharapkan untuk diikuti oleh angka, yang mewakili lebar tampilan maksimum dari output. Dan tentu saja, Anda mungkin memasukkan parameter lain ke dalam skrip Anda yang hanya berupa nilai data, yang sama sekali bukan opsi.

Untungnya getopts menangani kerumitan ini untuk Anda. Dan karena ini adalah bawaan, ini tersedia di semua sistem yang memiliki shell Bash, jadi tidak ada yang perlu diinstal.

Catatan: getopt Bukan getopt

Ada utilitas lama bernama getopt . Ini adalah program utilitas kecil, bukan bawaan. Ada banyak versi getopt yang berbeda dengan perilaku yang berbeda, sedangkan getops mengikuti pedoman POSIX.

 ketik getopt
 ketik getopt 

menggunakan perintah type untuk melihat perbedaan antara getop dan getops

Karena getopt bukan bawaan, getopt tidak membagikan beberapa manfaat otomatis yang dilakukan getopts , seperti menangani spasi dengan bijaksana. Dengan getopts , Bash Shell menjalankan skrip Anda dan Bash Shell melakukan penguraian opsi. Anda tidak perlu menjalankan program eksternal untuk menangani penguraian.

Imbalannya adalah getopts tidak menangani nama opsi format panjang putus-putus ganda. Jadi Anda dapat menggunakan opsi yang diformat seperti -w tetapi tidak ” ---wide-format .” Di sisi lain, jika Anda memiliki skrip yang menerima opsi -a , -b , dan -c , getopts memungkinkan Anda menggabungkannya seperti -abc , -bca , atau -bac dan seterusnya.

Kami sedang mendiskusikan dan mendemonstrasikan getopts di artikel ini, jadi pastikan Anda menambahkan "s" terakhir ke nama perintah.

TERKAIT: Cara Keluar dari Spasi di Jalur File di Baris Perintah Windows

Rekap Cepat: Menangani Nilai Parameter

Skrip ini tidak menggunakan opsi putus-putus seperti -a atau -b . Itu menerima parameter "normal" pada baris perintah dan ini diakses di dalam skrip sebagai nilai.

 #!/bin/bash

# dapatkan variabel satu per satu
echo "Variabel Satu: $1" 
echo "Variabel Dua: $2" 
echo "Variabel Tiga: $3"

# loop melalui variabel
untuk var di " $@" lakukan
gema "$ var" 
selesai

Parameter diakses di dalam skrip sebagai variabel $1 , $2 , atau $3 .

Salin teks ini ke editor dan simpan sebagai file bernama “variables.sh.” Kita harus membuatnya dapat dieksekusi dengan perintah chmod . Anda harus melakukan langkah ini untuk semua skrip yang kita diskusikan. Cukup ganti nama file skrip yang sesuai setiap kali.

 chmod +x variabel.sh 

menggunakan perintah chmod untuk membuat skrip dapat dieksekusi

Jika kami menjalankan skrip kami tanpa parameter, kami mendapatkan output ini.

 ./variables.sh 

menjalankan skrip tanpa parameter

Kami tidak melewati parameter sehingga skrip tidak memiliki nilai untuk dilaporkan. Mari kita berikan beberapa parameter kali ini.

 ./variables.sh bagaimana menjadi geek 

menjalankan skrip dengan tiga kata sebagai parameternya

Seperti yang diharapkan, variabel $1 , $2 , dan $3 telah disetel ke nilai parameter dan kita melihatnya dicetak.

Jenis penanganan parameter satu-untuk-satu ini berarti kita perlu mengetahui terlebih dahulu berapa banyak parameter yang akan ada. Loop di bagian bawah skrip tidak peduli berapa banyak parameter yang ada, selalu loop melalui semuanya.

Jika kami memberikan parameter keempat, itu tidak ditetapkan ke variabel, tetapi loop masih menanganinya.

 ./variables.sh cara membuat situs web geek 

meneruskan empat parameter ke skrip yang hanya dapat menangani tiga

Jika kita menempatkan tanda kutip di sekitar dua kata, mereka diperlakukan sebagai satu parameter.

 ./variables.sh bagaimana "menjadi geek" 

mengutip dua parameter baris perintah agar mereka diperlakukan sebagai satu parameter

Jika kita membutuhkan skrip untuk menangani semua kombinasi opsi, opsi dengan argumen, dan parameter tipe data "normal", kita perlu memisahkan opsi dari parameter reguler. Kita dapat mencapainya dengan menempatkan semua opsi—dengan atau tanpa argumen— sebelum parameter reguler.

Tapi jangan berlari sebelum kita bisa berjalan. Mari kita lihat kasus paling sederhana untuk menangani opsi baris perintah.

Opsi Penanganan

Kami menggunakan getopts dalam loop while . Setiap iterasi dari loop bekerja pada satu opsi yang diteruskan ke skrip. Dalam setiap kasus, variabel OPTION diatur ke opsi yang diidentifikasi oleh getopts .

Dengan setiap iterasi dari loop, getopts berpindah ke opsi berikutnya. Ketika tidak ada opsi lagi, getopts mengembalikan false dan loop while keluar.

Cara Menggunakan Pernyataan Kasus dalam Skrip Bash
TERKAIT Cara Menggunakan Pernyataan Kasus di Skrip Bash

Variabel OPTION dicocokkan dengan pola di setiap klausa pernyataan kasus. Karena kami menggunakan pernyataan kasus, tidak masalah urutan apa yang disediakan opsi pada baris perintah. Setiap opsi dijatuhkan ke dalam pernyataan kasus dan klausa yang sesuai dipicu.

Klausa individual dalam pernyataan kasus memudahkan untuk melakukan tindakan khusus opsi dalam skrip. Biasanya, dalam skrip dunia nyata, Anda akan menetapkan variabel di setiap klausa, dan ini akan bertindak sebagai flag lebih lanjut dalam skrip, mengizinkan atau menolak beberapa fungsi.

Salin teks ini ke editor dan simpan sebagai skrip yang disebut "options.sh", dan buat itu dapat dieksekusi.

 #!/bin/bash

sementara getopts 'abc' OPSI; melakukan
  kasus "$OPTION" di 
    sebuah) 
      echo "Pilihan yang digunakan" ;;

    b)
      echo "Opsi b digunakan"
      ;;

    c)
      echo "Opsi c digunakan"
      ;;

    ?) 
      echo "Penggunaan: $(nama dasar $0) [-a] [-b] [-c]"
      keluar 1
      ;;
  esac
selesai

Ini adalah garis yang mendefinisikan perulangan while.

 sementara getopts 'abc' OPSI; melakukan

Perintah getopts diikuti oleh string opsi . Ini mencantumkan huruf yang akan kita gunakan sebagai opsi. Hanya huruf dalam daftar ini yang dapat digunakan sebagai opsi. Jadi dalam hal ini, -d tidak valid. Ini akan terjebak oleh ?) klausa karena getopts mengembalikan tanda tanya “ ? ” untuk opsi yang tidak teridentifikasi. Jika itu terjadi, penggunaan yang benar dicetak ke jendela terminal:

 echo "Penggunaan: $(nama dasar $0) [-a] [-b] [-c]"

Menurut konvensi, membungkus opsi dalam tanda kurung “ [] ” dalam jenis pesan penggunaan yang benar ini berarti opsi tersebut opsional. Perintah basename menghapus semua jalur direktori dari nama file. Nama file skrip disimpan dalam $0 dalam skrip Bash.

Mari gunakan skrip ini dengan kombinasi baris perintah yang berbeda.

 ./options.sh -a
 ./options.sh -a -b -c
 ./options.sh -ab -c
 ./options.sh -cab 

menguji skrip yang dapat menerima opsi baris perintah tipe sakelar

Seperti yang dapat kita lihat, semua kombinasi opsi pengujian kami diuraikan dan ditangani dengan benar. Bagaimana jika kita mencoba opsi yang tidak ada?

 ./options.sh -d 

Opsi yang tidak dikenal dilaporkan oleh shell dan skrip

Klausa penggunaan dipicu, yang bagus, tetapi kami juga mendapatkan pesan kesalahan dari shell. Itu mungkin atau mungkin tidak masalah untuk kasus penggunaan Anda. Jika Anda memanggil skrip dari skrip lain yang harus mengurai pesan kesalahan, akan lebih sulit jika shell juga menghasilkan pesan kesalahan.

Mematikan pesan kesalahan shell sangat mudah. Yang perlu kita lakukan adalah meletakkan titik dua ” : ” sebagai karakter pertama dari string opsi.

Edit file "options.sh" Anda dan tambahkan titik dua sebagai karakter pertama dari string opsi, atau simpan skrip ini sebagai "options2.sh", dan buat itu dapat dieksekusi.

 #!/bin/bash

sementara getopts ':abc' OPSI; melakukan
  kasus "$OPTION" di 
    sebuah) 
      echo "Pilihan yang digunakan" 
      ;;

    b)
      echo "Opsi b digunakan"
      ;;

    c)
      echo "Opsi c digunakan"
      ;;

    ?) 
      echo "Penggunaan: $(nama dasar $0) [-a] [-b] [-c]"
      keluar 1
      ;;
  esac
selesai

Ketika kami menjalankan ini dan menghasilkan kesalahan, kami menerima pesan kesalahan kami sendiri tanpa pesan shell.

 ./options2.sh.sh -d 

Opsi yang tidak dikenal dilaporkan oleh skrip saja

Menggunakan getopts Dengan Argumen Opsi

Untuk memberi tahu getopts bahwa opsi akan diikuti oleh argumen, beri tanda titik dua ” : ” tepat di belakang huruf opsi dalam string opsi.

Jika kita mengikuti "b" dan "c" dalam string opsi kita dengan titik dua, getopt akan mengharapkan argumen untuk opsi ini. Salin skrip ini ke editor Anda dan simpan sebagai "arguments.sh", dan buat itu dapat dieksekusi.

Ingat, titik dua pertama dalam string opsi digunakan untuk menekan pesan kesalahan shell—ini tidak ada hubungannya dengan pemrosesan argumen.

Saat getopt memproses opsi dengan argumen, argumen ditempatkan di variabel OPTARG . Jika Anda ingin menggunakan nilai ini di tempat lain dalam skrip Anda, Anda harus menyalinnya ke variabel lain.

 #!/bin/bash

while getopts ':ab:c:' OPSI; melakukan

  kasus "$OPTION" di
    sebuah)
      echo "Pilihan yang digunakan"
      ;;

    b)
      argB="$OPTARG"
      echo "Opsi b digunakan dengan: $argB"
      ;;

    c)
      argC="$OPTARG"
      echo "Opsi c digunakan dengan: $argC"
      ;;

    ?)
      echo "Penggunaan: $(nama dasar $0) [-a] [-b argumen] [-c argumen]"
      keluar 1
      ;;
  esac

selesai

Mari kita jalankan itu dan lihat cara kerjanya.

 ./arguments.sh -a -b "cara geek" -c reviewgeek
 ./arguments.sh -c reviewgeek -a 

menguji skrip yang dapat menangani argumen opsi

Jadi sekarang kita dapat menangani opsi dengan atau tanpa argumen, terlepas dari urutan yang diberikan pada baris perintah.

Tapi bagaimana dengan parameter biasa? Kami katakan sebelumnya bahwa kami tahu kami harus meletakkannya di baris perintah setelah opsi apa pun. Mari kita lihat apa yang terjadi jika kita melakukannya.

Opsi dan Parameter Pencampuran

Kami akan mengubah skrip kami sebelumnya untuk memasukkan satu baris lagi. Ketika loop while telah keluar dan semua opsi telah ditangani, kami akan mencoba mengakses parameter reguler. Kami akan mencetak nilainya dalam $1 .

Simpan skrip ini sebagai "arguments2.sh", dan buat itu dapat dieksekusi.

 #!/bin/bash

while getopts ':ab:c:' OPSI; melakukan

  kasus "$OPTION" di
    sebuah)
      echo "Pilihan yang digunakan"
      ;;

    b)
      argB="$OPTARG"
      echo "Opsi b digunakan dengan: $argB"
      ;;

    c)
      argC="$OPTARG"
      echo "Opsi c digunakan dengan: $argC"
      ;;

    ?)
      echo "Penggunaan: $(nama dasar $0) [-a] [-b argumen] [-c argumen]"
      keluar 1
      ;;
  esac

selesai

echo "Variabel satu adalah: $1"

Sekarang kita akan mencoba beberapa kombinasi opsi dan parameter.

 ./arguments2.sh dave
 ./arguments2.sh -a dave
 ./arguments2.sh -a -c how-to-geek dave 

Gagal mengakses parameter standar dalam skrip yang menerima argumen opsi

Jadi sekarang kita bisa melihat masalahnya. Segera setelah opsi apa pun digunakan, variabel $1 dan seterusnya diisi dengan flag opsi dan argumennya. Dalam contoh terakhir, $4 akan menyimpan nilai parameter "dave", tetapi bagaimana Anda mengaksesnya di skrip Anda jika Anda tidak tahu berapa banyak opsi dan argumen yang akan digunakan?

Jawabannya adalah dengan menggunakan OPTIND dan perintah shift .

Perintah shift membuang parameter pertama—terlepas dari jenisnya—dari daftar parameter. Parameter lainnya “shuffle down”, sehingga parameter 2 menjadi parameter 1, parameter 3 menjadi parameter 2, dan seterusnya. Jadi $2 menjadi $1 , $3 menjadi $2 , dan seterusnya.

Jika Anda memberikan shift dengan angka, itu akan menghilangkan banyak parameter dari daftar.

OPTIND menghitung opsi dan argumen saat ditemukan dan diproses. Setelah semua opsi dan argumen diproses OPTIND akan menjadi satu lebih tinggi dari jumlah opsi. Jadi jika kita menggunakan parameter shift to trim (OPTIND-1) dari daftar parameter, kita akan dibiarkan dengan parameter reguler di $1 dan seterusnya.

Itulah tepatnya yang dilakukan skrip ini. Simpan skrip ini sebagai "arguments3.sh" dan buat itu dapat dieksekusi.

 #!/bin/bash

while getopts ':ab:c:' OPSI; melakukan
  kasus "$OPTION" di
    sebuah)
      echo "Pilihan yang digunakan"
      ;;

    b)
      argB="$OPTARG"
      echo "Opsi b digunakan dengan: $argB"
      ;;

    c)
      argC="$OPTARG"
      echo "Opsi c digunakan dengan: $argC"
      ;;

    ?)
      echo "Penggunaan: $(nama dasar $0) [-a] [-b argumen] [-c argumen]"
      keluar 1
      ;;
  esac
selesai

echo "Sebelum - variabel satu adalah: $1"
geser "$(($OPTIND -1))"
echo "Setelah - variabel satu adalah: $1"
echo "Argumen lainnya (operan)"

untuk x dalam "$@"
melakukan
  gema $x
selesai

Kami akan menjalankan ini dengan campuran opsi, argumen, dan parameter.

 ./arguments3.sh -a -c how-to-geek "dave dee" dozy paruh baya mick tich 

Mengakses parameter standar dengan benar dalam skrip yang menerima argumen opsi

Kita dapat melihat bahwa sebelum kita memanggil shift , $1 menahan “-a”, tetapi setelah perintah shift $1 menyimpan parameter non-option, non-argumen pertama kita. Kami dapat mengulang semua parameter semudah yang kami bisa dalam skrip tanpa penguraian opsi.

Itu Selalu Baik Untuk Memiliki Pilihan

Menangani opsi dan argumennya dalam skrip tidak perlu rumit. Dengan getopts Anda dapat membuat skrip yang menangani opsi baris perintah, argumen, dan parameter persis seperti skrip asli yang sesuai dengan POSIX.