আপনি যদি লিনাক্স বা ম্যাক ব্যবহার করেন, প্রতিবার আপনি একটি টার্মিনাল খুললে আপনি একটি শেল অ্যাপ্লিকেশন ব্যবহার করছেন৷
একটি শেল হল একটি ইন্টারফেস যা আপনাকে আপনার সিস্টেমে কমান্ড চালাতে সাহায্য করে।
শেল এনভায়রনমেন্ট ভেরিয়েবল হোস্ট করে এবং কমান্ড ইতিহাস এবং স্বয়ংক্রিয়-সম্পূর্ণতার মতো দরকারী বৈশিষ্ট্য রয়েছে।
আপনি যদি এমন ব্যক্তি হন যে জিনিসগুলি কীভাবে কাজ করে তা শিখতে পছন্দ করেন তবে এই পোস্টটি আপনার জন্য উপযুক্ত হবে!
একটি শেল কিভাবে কাজ করে?
আমাদের নিজস্ব শেল অ্যাপ্লিকেশন তৈরি করতে আসুন একটি শেল আসলে কী তা নিয়ে চিন্তা করি:
প্রথমে, একটি প্রম্পট থাকে, সাধারণত আপনার বর্তমান ব্যবহারকারী এবং বর্তমান ডিরেক্টরির মতো কিছু অতিরিক্ত তথ্য সহ, তারপর আপনি একটি কমান্ড টাইপ করেন এবং যখন আপনি এন্টার টিপুন তখন ফলাফলগুলি আপনার স্ক্রিনে প্রদর্শিত হয়৷
হ্যাঁ, এটি বেশ মৌলিক শোনাচ্ছে, কিন্তু এটি কি আপনাকে কিছু মনে করিয়ে দেয় না?
আপনি যদি pry
এর কথা ভাবছেন তাহলে আপনি ঠিক!
আপনার অপারেটিং সিস্টেমের জন্য মূলত একটি REPL (Read-Eval-Print-Loop) এর একটি শেল৷
জেনে যে আমরা আপনার শেলের প্রথম সংস্করণ লিখতে পারি :
prompt = "> " print prompt while (input = gets.chomp) break if input == "exit" system(input) print prompt end
এটি আমাদের একটি ন্যূনতম, কিন্তু কার্যকরী শেল দেবে। আমরা একটি লাইব্রেরি ব্যবহার করে এটিকে উন্নত করতে পারি যা অন্য অনেক REPL-এর মতো অ্যাপ্লিকেশন ব্যবহার করে।
সেই লাইব্রেরিটিকে Readline
বলা হয় .
রিডলাইন লাইব্রেরি ব্যবহার করা
রিডলাইন রুবি স্ট্যান্ডার্ড লাইব্রেরির অংশ, তাই ইনস্টল করার কিছু নেই, আপনাকে শুধু require
করতে হবে এটা।
Readline
ব্যবহার করার সুবিধাগুলির মধ্যে একটি এটি আমাদের জন্য স্বয়ংক্রিয়ভাবে একটি কমান্ড ইতিহাস রাখতে পারে।
এটি কমান্ড প্রম্পট এবং অন্যান্য অনেক কিছু প্রিন্ট করার যত্ন নিতে পারে৷
এখানে আমাদের শেলের v2 আছে, এবার Readline
ব্যবহার করে :
require 'readline' while input = Readline.readline("> ", true) break if input == "exit" system(input) end
এটি দুর্দান্ত, আমরা দুটি puts
থেকে মুক্তি পেয়েছি প্রম্পটের জন্য এবং এখন আমাদের কাছে Readline
থেকে কিছু শক্তিশালী ক্ষমতার অ্যাক্সেস আছে . উদাহরণস্বরূপ, আমরা একটি শব্দ (CTRL + W
মুছতে কীবোর্ড শর্টকাট ব্যবহার করতে পারি ) অথবা এমনকি ইতিহাস অনুসন্ধান করুন (CTRL + R
)!
পুরো ইতিহাস প্রিন্ট করতে একটি নতুন কমান্ড যোগ করা যাক:
require 'readline' while input = Readline.readline("> ", true) break if input == "exit" puts Readline::HISTORY.to_a if input == "hist" # Remove blank lines from history Readline::HISTORY.pop if input == "" system(input) end
মজার ঘটনা:আপনি যদি এই কোডটি pry-এ চেষ্টা করেন তবে আপনি pry-এর কমান্ড ইতিহাস পাবেন! কারণ হল যে pry এছাড়াও
Readline
ব্যবহার করছে , এবংReadline::HISTORY
ভাগ করা অবস্থা।
এখন আপনি hist
টাইপ করতে পারেন আপনার কমান্ড ইতিহাস পেতে 🙂
স্বয়ংক্রিয়-সম্পূর্ণতা যোগ করা হচ্ছে
আপনার প্রিয় শেলের স্বয়ংক্রিয়-সম্পূর্ণ বৈশিষ্ট্যের জন্য ধন্যবাদ আপনি প্রচুর টাইপিং সংরক্ষণ করতে সক্ষম হবেন। রিডলাইন আপনার শেলটিতে এই বৈশিষ্ট্যটি একত্রিত করা সত্যিই সহজ করে তোলে৷
আমাদের ইতিহাস থেকে স্বয়ংক্রিয়ভাবে সম্পূর্ণ করার কমান্ড দিয়ে শুরু করা যাক।
উদাহরণ :
comp = proc { |s| Readline::HISTORY.grep(/^#{Regexp.escape(s)}/) } Readline.completion_append_character = " " Readline.completion_proc = comp ## rest of the code goes here ##
এই কোডের সাহায্যে আপনি <tab>
টিপে পূর্বে টাইপ করা কমান্ডগুলি স্বয়ংক্রিয়ভাবে সম্পূর্ণ করতে সক্ষম হবেন চাবি. এখন আসুন এটিকে আরও এক ধাপ এগিয়ে নিয়ে যাই এবং ডিরেক্টরি স্বয়ংক্রিয়-সম্পূর্ণতা যোগ করি।
উদাহরণ :
comp = proc do |s| directory_list = Dir.glob("#{s}*") if directory_list.size > 0 directory_list else Readline::HISTORY.grep(/^#{Regexp.escape(s)}/) end end
completion_proc
সম্ভাব্য প্রার্থীদের তালিকা প্রদান করে, এই ক্ষেত্রে আমাদের কেবল Dir.glob
ব্যবহার করে টাইপ করা স্ট্রিংটি একটি ডিরেক্টরি নামের অংশ কিনা তা পরীক্ষা করতে হবে। . রিডলাইন বাকিটা দেখবে!
সিস্টেম পদ্ধতি বাস্তবায়ন করা
এখন আপনার কাছে ইতিহাস এবং স্বয়ংক্রিয়-সম্পূর্ণতা সহ একটি কার্যকরী শেল থাকা উচিত, 25 লাইন কোডের জন্য খুব খারাপ নয় 🙂
তবে এমন কিছু আছে যা আমি আরও গভীরে খনন করতে চাই, যাতে আপনি আসলে একটি আদেশ কার্যকর করার পর্দার পিছনে কী ঘটছে সে সম্পর্কে কিছু অন্তর্দৃষ্টি পেতে পারেন৷
এটি system
দ্বারা সম্পন্ন হয় পদ্ধতি, C তে এই পদ্ধতিটি আপনার কমান্ড পাঠায় /bin/sh
, যা একটি শেল অ্যাপ্লিকেশন। চলুন দেখি কিভাবে আপনি কি /bin/sh
বাস্তবায়ন করতে পারেন রুবিতে করে।
দ্রষ্টব্য :এটি শুধুমাত্র Linux / Mac 🙂
এ কাজ করবে৷
সিস্টেম পদ্ধতি:
def system(command) fork { exec(command) } end
এখানে যা ঘটে তা হল fork
বর্তমান প্রক্রিয়াটির একটি নতুন অনুলিপি তৈরি করে, তারপরে এই প্রক্রিয়াটি কমান্ড দ্বারা প্রতিস্থাপিত হয় যা আমরা exec
এর মাধ্যমে চালাতে চাই। পদ্ধতি এটি লিনাক্স প্রোগ্রামিং এর একটি খুব সাধারণ প্যাটার্ন।
আপনি যদি কাঁটাচামচ না করেন তবে বর্তমান প্রক্রিয়াটি প্রতিস্থাপিত হয়, যার অর্থ হল আপনি যখন কমান্ডটি চালাচ্ছেন (ls
, cd
বা অন্য কিছু) হয়ে গেলে আপনার রুবি প্রোগ্রাম এটির সাথে শেষ হয়ে যাবে।
আপনি এখানে ঘটছে দেখতে পারেন:
def system(command) exec(command) end system('ls') # This code will never run! puts "after system"
উপসংহার
এই পোস্টে আপনি শিখেছেন যে একটি শেল হল একটি REPL-এর মতো ইন্টারফেস (ভাবুন irb
/ pry
) আপনার সিস্টেমের সাথে ইন্টারঅ্যাক্ট করার জন্য। আপনি শক্তিশালী Readline
ব্যবহার করে কীভাবে নিজের শেল তৈরি করবেন তাও শিখেছেন লাইব্রেরি, যা ইতিহাস এবং স্বয়ংসম্পূর্ণতার মতো অনেক বিল্ট-ইন বৈশিষ্ট্য সরবরাহ করে (কিন্তু এটি কীভাবে কাজ করে তা আপনাকে সংজ্ঞায়িত করতে হবে)।
এবং তার পরে আপনি fork
সম্পর্কে শিখেছেন + exec
লিনাক্স প্রোগ্রামিং প্রকল্পে সাধারণত ব্যবহৃত প্যাটার্ন।
আপনি যদি এই পোস্টটি উপভোগ করেন তবে আপনি কি আমাকে একটি উপকার করতে পারেন এবং আপনার সমস্ত রুবি বন্ধুদের সাথে শেয়ার করতে পারেন? এটি ব্লগের বৃদ্ধিতে সাহায্য করবে এবং আরও বেশি লোক শিখতে পারবে 🙂