Compare commits

..

9 Commits
lab2 ... main

BIN
.DS_Store vendored

Binary file not shown.

@ -9,6 +9,7 @@ Abstract:
Content: Content:
[Lab 1: Hello world, user interface and number funcations](https://git.djft.ru/darius-atlas/kubsu-sm5-ruby/src/branch/main/lab1) [Lab 1: Hello world, user interface and number funcations](https://git.djft.ru/darius-atlas/kubsu-sm5-ruby/src/branch/main/lab1)
[Lab 2: Person classes](https://git.djft.ru/darius-atlas/kubsu-sm5-ruby/src/branch/main/lab2)
--- ---

@ -0,0 +1,102 @@
# Lab 1
---
## 01. Hello world
Task:
Установить компилятор и текстовый редактор. Реализовать и
вызвать Hello World c комментарием. (ну как всегда)
```bash
ruby 01_hello_world.rb
```
Returns:
```
Hello, world!
```
## 02. User interface
Task:
Принять имя пользователя как аргумент программы.
Поздороваться с пользователем с использованием форматирования
строки. Спросить какой язык у пользователя любимый, в случае, если это
ruby, ответить что пользователь подлиза, иначе обязательно ответить, что
скоро будет ruby и поставить различные комментарии для нескольких
языков.
```bash
ruby 02_user_interface.rb darius
```
## 03. User interface with ruby and os commands execution
Task:
3.Продолжение предыдущего задания. Попросить пользователя
ввести команду языка ruby. И команду OC. Выполнить команду руби и
команду операционной системы.
```bash
ruby 03_fork_02_with_exec_pasted_command.rb darius
```
## 04. Number funcations
Task:
«Работа с числами». Составить 3 метода для работы с цифрами или делителей числа на основании варианта. Каждый метод отдельный коммит.
- Метод 1. Найти количество четных чисел, не взаимно простых с данным
- Метод 2. Найти максимальную цифры числа, не делящуюся на 3.
- Метод 3. Найти произведение максимального числа, не взаимно простого с данным, не делящегося на наименьший делитель исходно числа, и суммы цифр числа, меньших 5.
```bash
ruby 04_number_funcs.rb
```
## 05. Number items operations
Task:
Написать методы, которые находят минимальный, элементы,
номер первого положительного элемента. Каждая операция в отдельном
методе. Решить задачу с помощью циклов(for и while).
```bash
ruby 05_number_items_operations.rb
```
## 06. Command line file arguments
Task:
Написать программу, которая принимает как аргумент два
значения. Первое значение говорит, какой из методов задачи 1
выполнить, второй говорит о том, откуда читать список аргументом
должен быть написан адрес файла. Далее необходимо прочитать массив
и выполнить метод.
```bash
ruby 06_command_line_file_arguments.rb min numbers.txt
```
```bash
ruby 06_command_line_file_arguments.rb first_positive numbers.txt
```
```bash
ruby 06_command_line_file_arguments.rb first_positive_index numbers.txt
```
---
Author: Artem-Darius Weber
Licence: MIT

@ -0,0 +1,16 @@
require_relative '../src/02_user_interface.rb'
RSpec.describe "Main" do
before do
allow(STDIN).to receive(:gets).and_return("ruby\n")
end
it "greets the user and checks Ruby as language" do
expect { main() }.to output("Hello my catgirl test_user! \nWhat is your love language?\nruby\nc++\npy\nПодлиза \n").to_stdout
end
it "handles unknown language input" do
allow(STDIN).to receive(:gets).and_return("java\n")
expect { main() }.to output(/Неизвестный язык: java/).to_stdout
end
end

@ -0,0 +1,34 @@
require_relative '../src/03_fork_02_with_exec_pasted_command.rb'
RSpec.describe "Main" do
before do
allow(STDIN).to receive(:gets).and_return("ruby\n", "puts 'Hello from Ruby!'\n", "echo 'Hello from shell!'\n")
allow(ARGV).to receive(:[]).with(0).and_return("test_user")
end
it "greets the user and processes ruby input" do
expect { main() }.to output(/Hello my catgirl test_user! \nWhat is your love language?\nruby\nc++\npy\одлиза \n/).to_stdout
end
it "executes valid ruby command" do
allow(STDIN).to receive(:gets).and_return("ruby\n", "puts 'Hello from Ruby!'\n")
expect { main() }.to output(/Hello from Ruby!/).to_stdout
end
it "handles ruby command execution error" do
allow(STDIN).to receive(:gets).and_return("ruby\n", "invalid_ruby_code\n")
expect { main() }.to output(/Ошибка выполнения команды Ruby: undefined local variable or method `invalid_ruby_code'/).to_stdout
end
it "executes shell command" do
allow(STDIN).to receive(:gets).and_return("ruby\n", "puts 'Hello from Ruby!'\n", "echo 'Hello from shell!'\n")
expect(main()).to include("Hello from shell!")
end
it "returns correct output for unknown programming language" do
allow(STDIN).to receive(:gets).and_return("java\n")
expect { main() }.to output(/Неизвестный язык: java/).to_stdout
end
end

@ -0,0 +1,2 @@
puts "Hello, world!"

@ -0,0 +1,30 @@
#!/usr/bin/env ruby
LANGUAGES = [
"ruby",
"c++",
"py"
]
def main()
user_name = ARGV[0]
puts "Hello my catgirl #{user_name}! \nWhat is your love language?"
LANGUAGES.each { |language| puts "#{language}", " " }
user_lang = STDIN.gets.chomp
case user_lang
when "ruby"
puts "Подлиза \n"
when "c++"
puts "MATLAB скушал? \n"
when "TS/JS"
puts "Фронтендер, Фууу! \n"
when "py"
puts "Девопсер, иди ДАГИ писать \n"
else
puts "Неизвестный язык: #{user_lang}"
end
end
main()

@ -0,0 +1,39 @@
#!/usr/bin/env ruby
LANGUAGES = [
"ruby",
"c++",
"py"
]
def main()
user_name = ARGV[0]
puts "Hello my catgirl #{user_name}! \nWhat is your love language?"
LANGUAGES.each { |language| puts "#{language}", " " }
user_lang = STDIN.gets.chomp
case user_lang
when "ruby"
puts "Подлиза \n"
when "c++"
puts "MATLAB скушал? \n"
when "TS/JS"
puts "Фронтендер, Фууу! \n"
when "py"
puts "Девопсер, иди ДАГИ писать \n"
end
puts "Введите команду на языке Ruby для выполнения:"
ruby_command = STDIN.gets.chomp
begin
eval(ruby_command)
rescue Exception => e
puts "Ошибка выполнения команды Ruby: #{e.message}"
end
puts "Введите команду операционной системы для выполнения:"
os_command = STDIN.gets.chomp
system(os_command)
end
main()

@ -0,0 +1,67 @@
require 'prime'
# @param n [Integer] Число, для которого нужно найти количество четных, не взаимно простых чисел.
# @return [Integer] Количество четных чисел, не взаимно простых с данным числом.
# @example
# count_even_non_coprimes(30) # => 14
def count_even_non_coprimes(n)
(1...n).count do |num|
num.even? && n.gcd(num) != 1
end
end
# @param n [Integer] Число, цифры которого нужно проверить.
# @return [Integer, nil] Максимальная цифра числа, не делящаяся на 3, или nil, если таких цифр нет.
# @example
# max_digit_not_divisible_by_three(483726) # => 8
def max_digit_not_divisible_by_three(n)
n.digits.select { |digit| digit % 3 != 0 }.max
end
# @param n [Integer] Число, для которого нужно найти наименьший делитель.
# @return [Integer] Наименьший делитель числа.
# @example
# smallest_divisor(30) # => 2
def smallest_divisor(n)
(2..n).find { |i| n % i == 0 }
end
# @param n [Integer] Число, для которого нужно найти подходящее максимальное число.
# @return [Integer, nil] Максимальное число, не взаимно простое с данным, не делящееся на наименьший делитель, или nil, если таких чисел нет.
# @example
# max_non_coprime_not_divisible_by_smallest_divisor(30) # => 28
def max_non_coprime_not_divisible_by_smallest_divisor(n)
divisor = smallest_divisor(n)
(1...n).select { |num| n.gcd(num) != 1 && num % divisor != 0 }.max
end
# @param n [Integer] Число, цифры которого нужно просуммировать.
# @return [Integer] Сумма цифр числа, меньших 5.
# @example
# sum_of_digits_less_than_five(483726) # => 9
def sum_of_digits_less_than_five(n)
n.digits.select { |digit| digit < 5 }.sum
end
# @param n [Integer] Число, для которого проводится вычисление.
# @return [Integer] Произведение максимального числа и суммы цифр числа.
# @example
# product_of_max_and_sum(30) # => 252
def product_of_max_and_sum(n)
max_number = max_non_coprime_not_divisible_by_smallest_divisor(n)
sum_digits = sum_of_digits_less_than_five(n)
max_number * sum_digits
end
puts count_even_non_coprimes(30) # => 14
puts max_digit_not_divisible_by_three(483726) # => 8
puts product_of_max_and_sum(30) # => 252

@ -0,0 +1,68 @@
def find_min_element_for(arr)
min_element = arr[0]
for element in arr
min_element = element if element < min_element
end
min_element
end
def find_min_element_while(arr)
min_element = arr[0]
index = 0
while index < arr.size
min_element = arr[index] if arr[index] < min_element
index += 1
end
min_element
end
def find_first_positive_index_for(arr)
for index in 0...arr.size
return index if arr[index] > 0
end
nil
end
def find_first_positive_index_while(arr)
index = 0
while index < arr.size
return index if arr[index] > 0
index += 1
end
nil
end
def find_first_positive_for(arr)
for element in arr
return element if element > 0
end
nil
end
def find_first_positive_while(arr)
index = 0
while index < arr.size
return arr[index] if arr[index] > 0
index += 1
end
nil
end
# INPUT
array = [-10, -5, 0, 3, 5, -2]
puts "Минимальный элемент (for): #{find_min_element_for(array)}"
puts "Минимальный элемент (while): #{find_min_element_while(array)}"
puts "Индекс первого положительного элемента (for): #{find_first_positive_index_for(array)}"
puts "Индекс первого положительного элемента (while): #{find_first_positive_index_while(array)}"
puts "Первый положительный элемент (for): #{find_first_positive_for(array)}"
puts "Первый положительный элемент (while): #{find_first_positive_while(array)}"

@ -0,0 +1,62 @@
def find_min_element(arr)
min_element = arr[0]
arr.each do |element|
min_element = element if element < min_element
end
min_element
end
def find_first_positive_index(arr)
arr.each_with_index do |element, index|
return index if element > 0
end
nil
end
def find_first_positive(arr)
arr.each do |element|
return element if element > 0
end
nil
end
def main
method_name = ARGV[0]
file_path = ARGV[1]
if method_name.nil? || file_path.nil?
puts "Неправльный формат ввода. Использование: ruby program.rb <method_name> <file_path>"
puts "method_name: 'min', 'first_positive_index' или 'first_positive'"
exit
end
begin
array = File.read(file_path).split.map(&:to_i)
rescue Errno::ENOENT
puts "Файл не найден: #{file_path}"
exit
end
case method_name
when 'min'
result = find_min_element(array)
puts "Минимальный элемент: #{result}"
when 'first_positive_index'
result = find_first_positive_index(array)
puts "Индекс первого положительного элемента: #{result.nil? ? 'Нет положительного элемента' : result}"
when 'first_positive'
result = find_first_positive(array)
puts "Первый положительный элемент: #{result.nil? ? 'Нет положительного элемента' : result}"
else
puts "Неизвестный метод: #{method_name}"
puts "Доступные методы: 'min', 'first_positive_index', 'first_positive'"
end
end
main if __FILE__ == $0

@ -0,0 +1 @@
-10 -5 0 3 5 -2

@ -1,88 +1,4 @@
# Lab 2 # Lab 2
> "И тут я обнаружил что случайно сделал 3-5 задачи в 1-2"
## Диаграмма классов:
```mermaid
classDiagram
class BinarySearchTree {
- root : Node
+ add(student : Student) : void
+ each(&block) : void
- insert(node : Node, student : Student) : Node
- in_order_traversal(node : Node, &block) : void
}
class Node {
- student : Student
- left : Node
- right : Node
+ Node(student : Student)
}
class Person {
- id : String
- git : String
- phone : String?
- telegram : String?
- email : String?
+ Person(id : String, git : String, phone: String, telegram: String, email: String)
+ phone=(String) : Boolean
+ telegram=(String) : Boolean
+ email=(String) : Boolean
+ surname_initials() : NotImplementedError
+ valid_phone_number() : Boolean <<class>>
+ valid_telegram() : Boolean <<class>>
+ valid_email() : Boolean <<class>>
+ git_present() : Boolean
+ contact_present() : Boolean
+ get_first_contact() : String
+ git=(git : String) : void
+ validate_id(id: String) : void <<class>>
+ validate_git(git : String) : void <<class>>
- valid_git?(git : String) : Boolean
- valid_id?(id : String) : Boolean
}
class StudentRepository {
+ read_from_txt(file_path : String) : List~Student~ <<class>>
+ write_to_txt(file_path : String, students : List~Student~) <<class>>
}
class StudentShort {
- surname_initials : String
+ get_surname_initials() : String
+ StudentShort(id : String, surname_initials : String, phone: String, telegram: String, email: String)
+ from_student(student : Student) : StudentShort <<class>>
+ from_string(id : String, info_string : String) : StudentShort <<class>>
+ to_s() : String
- parse_contact_string(contact_string: String) : Array<String> <<class>>
}
class Student {
- surname : String
- name : String
- patronymic : String
- birth_date : Date
- const NAME_REGEX : String
+ Student(id : String, git : String, phone: String, telegram: String, email: String, surname : String, name : String, patronymic : String, birth_date : Date)
+ from_string(student_string : String) : Student <<class>>
+ surname_initials() : String
+ to_s() : String
+ get_info() : String
+ surname=(surname : String)
+ name=(name : String)
+ patronymic=(patronymic : String)
+ birth_date=(birthdate : String)
+ valid_name?(name : String) : Boolean <<class>>
- name_initial(name : String) : String
- patronymic(patronymic : String) : String
}
BinarySearchTree o-- Node
Node o-- Student
Person <|-- Student
Person <|-- StudentShort
Student <.. StudentRepository
```

@ -1,46 +0,0 @@
class BinarySearchTree
include Enumerable
class Node
attr_accessor :student, :left, :right
def initialize(student)
@student = student
@left = nil
@right = nil
end
end
def initialize
@root = nil
end
def add(student)
@root = insert(@root, student)
end
def each(&block)
in_order_traversal(@root, &block)
end
private
def insert(node, student)
return Node.new(student) if node.nil?
if student.birth_date < node.student.birth_date
node.left = insert(node.left, student)
else
node.right = insert(node.right, student)
end
node
end
def in_order_traversal(node, &block)
return if node.nil?
in_order_traversal(node.left, &block)
yield node.student
in_order_traversal(node.right, &block)
end
end

@ -1,92 +1,59 @@
require 'date'
require_relative 'person'
require_relative 'student' require_relative 'student'
require_relative 'student_short' require_relative 'student_short'
require_relative 'student_repository'
require_relative 'binary_search_tree'
def test_classes begin
person = Person.new(id: '1', git: 'https://github.com/example', phone: '+798912465', telegram: '@example_user', email: 'test@example.com') student_string = "Норакет Норакет Норакет | ID: 2 | Phone: +1234567890 | Telegram: @noracat | Email: nora@example.com | Git: https://github.com/nora"
puts "Person Contact Info: #{person.get_first_contact}" student_from_string = Student.from_string(student_string)
puts "Создан объект из строки:\n#{student_from_string}"
student = Student.new(
id: '2',
git: 'https://github.com/student_example', student1 = Student.new(
surname: 'Иванов', surname: 'Алексеевич',
name: 'Иван', name: 'Артем-Дариус',
patronymic: 'Иванович', patronymic: 'Вебер',
birth_date: Date.new(2000, 5, 15), id: 1,
phone: '+79891234567', git: 'https://github.com/space-creator'
telegram: '@student_ivan', )
email: 'ivanov@example.com'
) student1.set_contacts(
puts "Student Full Info: #{student.to_s}" phone: '+79891242223',
puts "Student Initials: #{student.surname_initials}" telegram: '@alstroemeria22',
puts "Student Contact Info: #{student.get_first_contact}" email: 'no-replay@djft.ru'
)
student_short = StudentShort.from_student(student)
puts "Student Short Info: #{student_short.to_s}" puts student1.get_info
puts "Surname and Initials: #{student1.surname_and_initials}"
file_path = 'students.txt' puts "Git Info: #{student1.git_info}"
students = StudentRepository.read_from_txt(file_path) puts "Contact Info: #{student1.contact_info}"
puts "Read Students from File:"
students.each { |s| puts s.to_s } student2 = Student.new(
surname: 'Норакет',
output_path = 'output_students.txt' name: 'Норакет',
StudentRepository.write_to_txt(output_path, students) patronymic: 'Фамилия'
puts "Students written to file: #{output_path}" )
end
student2.set_contacts(
def test_parsing phone: '+70000000000'
student_string = 'Иванов Иван Иванович | ID: 3 | Phone: +79876543210 | Telegram: @ivan_user | Email: ivan@example.com | Git: https://github.com/ivanov | Birth Date: 2001-01-01' )
student = Student.from_string(student_string)
puts "Parsed Student: #{student.to_s}" puts student1
puts '-' * 40
short_string = 'Иванов И.И., Contact: Phone: +79876543210' puts student2
student_short = StudentShort.from_string('4', short_string)
puts "Parsed StudentShort: #{student_short.to_s}"
end student_short_from_student = StudentShort.new(student1)
puts "StudentShort from Student object:"
def test_binary_search_tree puts "ID: #{student_short_from_student.id}"
bst = BinarySearchTree.new puts "Surname and Initials: #{student_short_from_student.surname_initials}"
puts "Git: #{student_short_from_student.git}"
student1 = Student.new( puts "Contact: #{student_short_from_student.contact}"
id: '1',
git: 'https://github.com/student1', student_short_from_string = StudentShort.from_string(5, 'Skye A.A., Git: https://github.com/skye, Contact: Phone: +4923467890')
surname: 'Смирнов', puts "StudentShort from string:"
name: 'Алексей', puts "ID: #{student_short_from_string.id}"
patronymic: 'Иванович', puts "Surname and Initials: #{student_short_from_string.surname_initials}"
birth_date: Date.new(1999, 2, 15), puts "Git: #{student_short_from_string.git}"
phone: '+123456789', puts "Contact: #{student_short_from_string.contact}"
telegram: '@student1', rescue ArgumentError => e
email: 'student1@example.com' puts "Ошибка: #{e.message}"
)
student2 = Student.new(
id: '2',
git: 'https://github.com/student2',
surname: 'Иванов',
name: 'Иван',
patronymic: 'Сергеевич',
birth_date: Date.new(2001, 5, 10),
phone: '+987654321',
telegram: '@student2',
email: 'student2@example.com'
)
bst.add(student1)
bst.add(student2)
puts "Students in BST (sorted by birth_date):"
bst.each { |student| puts student.to_s }
end
if __FILE__ == $0
puts "=== Testing Classes ==="
test_classes
puts "\n=== Testing Parsing ==="
test_parsing
puts "\n=== Testing Binary Search Tree ==="
test_binary_search_tree
end end

@ -1,85 +1,65 @@
class Person class Person
attr_reader :id, :git, :phone, :telegram, :email attr_accessor :id, :git
def initialize(id:, git: nil, phone: nil, telegram: nil, email: nil) def initialize(args = {})
self.class.validate_id(id) @id = args[:id] || nil
self.phone = phone @git = args[:git]
self.telegram = telegram validate
self.email = email end
raise ArgumentError, "Необходимо указать хотя бы один контакт (телефон, Telegram или email)" unless contact_present?
@id = id
@git = git
end
def surname_initials
raise NotImplementedError, "#{self.class} must implement the 'surname_initials' method"
end
def git_present? def set_contacts(phone: nil, telegram: nil, email: nil)
!@git.nil? && !@git.empty? @phone = phone
end raise ArgumentError, "Invalid phone number format: #{@phone}" if @phone && !self.class.valid_phone_number?(@phone)
def contact_present? @telegram = telegram
@phone || @telegram || @email raise ArgumentError, "Invalid telegram format: #{@telegram}" if @telegram && !self.class.valid_telegram?(@telegram)
end
def get_first_contact @email = email
[@phone, @telegram, @email].compact.first raise ArgumentError, "Invalid email format: #{@email}" if @email && !self.class.valid_email?(@email)
end end
def self.valid_phone_number?(phone) def self.valid_phone_number?(phone)
/\A\+?[0-9]{9,15}\z/.match?(phone) phone.match?(/\A\+?[0-9]{10,15}\z/)
end end
def self.valid_telegram?(telegram) def self.valid_name?(name)
/\A@[A-Za-z0-9_]{5,32}\z/.match?(telegram) name.match?(/\A[А-Яа-яЁёA-Za-z\-]+\z/)
end end
def self.valid_email?(email) def self.valid_telegram?(telegram)
/\A[^@\s]+@[^@\s]+\.[^@\s]+\z/.match?(email) telegram.nil? || telegram.match?(/\A@[A-Za-z0-9_]{5,32}\z/)
end end
def phone=(phone) def self.valid_email?(email)
if phone && !self.class.valid_phone_number?(phone) email.nil? || email.match?(/\A[^@\s]+@[^@\s]+\.[^@\s]+\z/)
raise ArgumentError, "Недопустимый номер телефона: #{phone}"
end end
@phone = phone
end
def telegram=(telegram) def self.valid_git?(git)
if telegram && !self.class.valid_telegram?(telegram) git.nil? || git.match?(/\Ahttps:\/\/github\.com\/[A-Za-z0-9_\-]+\z/)
raise ArgumentError, "Некорректный Telegram: #{telegram}"
end end
@telegram = telegram
end
def email=(email) def git_present?
if email && !self.class.valid_email?(email) !@git.nil? && !@git.empty?
raise ArgumentError, "Некорректный email: #{email}"
end end
@email = email
end
def git=(git) def contact_present?
self.class.validate_git(git) !(@phone.nil? || @phone.empty?) || !(@telegram.nil? || @telegram.empty?) || !(@email.nil? || @email.empty?)
@git = git end
end
def self.validate_id(id) def validate
raise ArgumentError, 'ID is required and must be a non-empty string' unless valid_id?(id) raise ArgumentError, "Git link is required" unless git_present?
end raise ArgumentError, "At least one contact (phone, telegram, or email) is required" unless contact_present?
end
def self.validate_git(git) def contact_info
raise ArgumentError, 'Git link is required' if git.nil? || git.strip.empty? return "Phone: #{@phone}" if @phone
raise ArgumentError, 'Invalid Git link format' unless valid_git?(git) return "Telegram: #{@telegram}" if @telegram
end return "Email: #{@email}" if @email
'No contact available'
end
def self.valid_git?(git) private
/\Ahttps:\/\/github\.com\/[A-Za-z0-9_\-]+\z/.match?(git)
end
def self.valid_id?(id) attr_reader :phone, :telegram, :email
id.is_a?(String) && !id.strip.empty? attr_writer :phone, :telegram, :email
end
end end

@ -1,103 +1,92 @@
require_relative 'person' require_relative 'person'
class Student < Person class Student < Person
attr_accessor :surname, :name, :patronymic, :birth_date attr_accessor :surname, :name, :patronymic
private def initialize(args = {})
super(args)
NAME_REGEX = /\A[А-Яа-яЁёA-Za-z\-]+\z/ @surname = args.fetch(:surname)
raise ArgumentError, "Invalid surname format: #{@surname}" unless self.class.valid_name?(@surname)
public
@name = args.fetch(:name)
def initialize(id:, git:, surname:, name:, patronymic:, birth_date:, phone: nil, telegram: nil, email: nil) raise ArgumentError, "Invalid name format: #{@name}" unless self.class.valid_name?(@name)
super(id: id, git: git, phone: phone, telegram: telegram, email: email)
self.surname = surname @patronymic = args.fetch(:patronymic)
self.name = name raise ArgumentError, "Invalid patronymic format: #{@patronymic}" unless self.class.valid_name?(@patronymic)
self.patronymic = patronymic
self.birth_date = birth_date set_contacts(
end phone: args[:phone],
telegram: args[:telegram],
def self.from_string(student_string) email: args[:email]
parts = student_string.split('|').map(&:strip) )
raise ArgumentError, 'Invalid student string format' if parts.size < 7 end
name_parts = parts[0].split(' ') def self.from_string(student_string)
raise ArgumentError, 'Invalid name format' if name_parts.size != 3 parts = student_string.split('|').map(&:strip)
surname, name, patronymic = parts[0].split(' ')
surname, name, patronymic = name_parts id = parts[1].split(': ').last.to_i
id = parts[1].split(': ').last phone = parts[2].split(': ').last
phone = parts[2].split(': ').last telegram = parts[3].split(': ').last
telegram = parts[3].split(': ').last email = parts[4].split(': ').last
email = parts[4].split(': ').last git = parts[5].split(': ').last
git = parts[5].split(': ').last
birth_date = Date.parse(parts[6].split(': ').last) new(
surname: surname,
new( name: name,
id: id, patronymic: patronymic,
git: git, id: id,
surname: surname, phone: phone,
name: name, telegram: telegram,
patronymic: patronymic, email: email,
birth_date: birth_date, git: git
phone: phone, )
telegram: telegram, end
email: email
) def self.read_from_txt(file_path)
end raise IOError, "File path is invalid or file does not exist: #{file_path}" unless File.exist?(file_path)
def surname_initials students = []
"#{@surname} #{name_initial(@name)}.#{patronymic_initial(@patronymic)}."
end File.foreach(file_path) do |line|
line.strip!
def to_s next if line.empty?
"#{@surname} #{@name} #{@patronymic} | ID: #{@id} | " \
"Phone: #{@phone || 'N/A'} | Telegram: #{@telegram || 'N/A'} | " \ begin
"Email: #{@email || 'N/A'} | Git: #{@git} | Birth Date: #{@birth_date}" student = from_string(line)
end students << student
rescue ArgumentError => e
def get_info puts "Error processing line: '#{line}'. Reason: #{e.message}"
"#{surname_initials}, Git: #{@git}, Contact: #{get_first_contact}, Birth Date: #{@birth_date}" end
end end
def surname=(surname) students
raise ArgumentError, 'Surname is required' if surname.nil? || surname.strip.empty? end
raise ArgumentError, "Invalid surname format: #{surname}" unless self.class.valid_name?(surname)
def self.write_to_txt(file_path, students)
@surname = surname raise ArgumentError, "Expected an array of Student objects" unless students.is_a?(Array) && students.all? { |s| s.is_a?(Student) }
end
File.open(file_path, 'w') do |file|
def name=(name) students.each do |student|
raise ArgumentError, 'Name is required' if name.nil? || name.strip.empty? file.puts student.to_s
raise ArgumentError, "Invalid name format: #{name}" unless self.class.valid_name?(name) end
end
@name = name
end puts "Data successfully written to #{file_path}"
rescue IOError => e
def patronymic=(patronymic) puts "An error occurred while writing to the file: #{e.message}"
raise ArgumentError, 'Patronymic is required' if patronymic.nil? || patronymic.strip.empty? end
raise ArgumentError, "Invalid patronymic format: #{patronymic}" unless self.class.valid_name?(patronymic)
def surname_and_initials
@patronymic = patronymic "#{@surname} #{name[0]}.#{patronymic[0]}."
end end
def birth_date=(birth_date) def to_s
raise ArgumentError, 'Birth date is required' if birth_date.nil? "#{@surname} #{@name} #{@patronymic} | ID: #{@id || 'N/A'} | " \
raise ArgumentError, "Invalid birth date: #{birth_date}" unless birth_date.is_a?(Date) "Phone: #{@phone || 'N/A'} | Telegram: #{@telegram || 'N/A'} | " \
"Email: #{@email || 'N/A'} | Git: #{@git || 'N/A'}"
@birth_date = birth_date end
end
def get_info
def self.valid_name?(name) "#{surname_and_initials}, Git: #{git_info}, Contact: #{contact_info}"
NAME_REGEX.match?(name) end
end
private
def name_initial(name)
name[0].upcase
end
def patronymic_initial(patronymic)
patronymic[0].upcase
end
end end

@ -1,39 +0,0 @@
require_relative 'student'
class StudentRepository
def self.read_from_txt(file_path)
raise IOError, "File does not exist: #{file_path}" unless File.exist?(file_path)
students = []
File.foreach(file_path) do |line|
line.strip!
next if line.empty?
begin
student = Student.from_string(line)
students << student
rescue ArgumentError => e
puts "Error processing line: '#{line}'. Reason: #{e.message}"
end
end
students
end
def self.write_to_txt(file_path, students)
unless students.is_a?(Array) && students.all? { |s| s.is_a?(Student) }
raise ArgumentError, 'Expected an array of Student objects'
end
File.open(file_path, 'w') do |file|
students.each do |student|
file.puts student.to_s
end
end
puts "Data successfully written to #{file_path}"
rescue IOError => e
puts "An error occurred while writing to the file: #{e.message}"
end
end

@ -1,59 +1,31 @@
require_relative 'person' require_relative 'person'
class StudentShort < Person class StudentShort < Person
attr_reader :surname_initials attr_reader :surname_initials, :contact
def initialize(id:, surname_initials:, phone: nil, telegram: nil, email: nil) def initialize(student)
super(id: id, phone: phone, telegram: telegram, email: email) super(id: student.id, git: student.git_info)
@surname_initials = surname_initials @surname_initials = student.surname_and_initials
freeze @contact = student.contact_info
end end
def self.from_student(student)
new(
id: student.id,
surname_initials: student.surname_initials,
phone: student.phone,
telegram: student.telegram,
email: student.email
)
end
def self.from_string(id, info_string)
parts = info_string.split(',').map(&:strip)
raise ArgumentError, 'Invalid info string format' if parts.size < 2
surname_initials = parts[0]
contact_string = parts[1].split(': ', 2).last.strip
phone, telegram, email = parse_contact_string(contact_string)
new( def self.from_string(id, info_string)
id: id, parts = info_string.split(', ').map(&:strip)
surname_initials: surname_initials, surname_initials = parts[0]
phone: phone, git = parts[1].split(': ').last
telegram: telegram, contact = parts[2].split(': ').last
email: email
)
end
def to_s new_instance = allocate
contact_info = get_first_contact() new_instance.send(:initialize_from_data, id, surname_initials, git, contact)
"#{@surname_initials}, Contact: #{contact_info}" new_instance
end end
private private
def self.parse_contact_string(contact_string) def initialize_from_data(id, surname_initials, git, contact)
case contact_string @id = id
when /\APhone: (.+)\z/i @surname_initials = surname_initials
[Regexp.last_match(1).strip, nil, nil] @git = git
when /\ATelegram: (.+)\z/i @contact = contact
[nil, Regexp.last_match(1).strip, nil]
when /\AEmail: (.+)\z/i
[nil, nil, Regexp.last_match(1).strip]
else
raise ArgumentError, "Invalid contact string format: #{contact_string}"
end end
end
end end

@ -1,3 +0,0 @@
Иванов Иван Иванович | ID: 1 | Phone: +12345678901 | Telegram: @ivanov_user | Email: ivanov@example.com | Git: https://github.com/ivanov
Петров Петр Петрович | ID: 2 | Phone: +98765432101 | Telegram: @petrov_user | Email: petrov@example.com | Git: https://github.com/petrov
Сидоров Сидор Сидорович | ID: 3 | Phone: +56789012345 | Telegram: @sidorov_user | Email: sidorov@example.com | Git: https://github.com/sidorov
Loading…
Cancel
Save