Загрузка данных
#!/usr/bin/env python3
import sys
import os
STATS_FILE = "chown_stats.txt"
# --- ПОЛНОЕ СООТВЕТСТВИЕ ВАРИАНТУ 13: СТРУКТУРА ДАННЫХ И ФУНКЦИИ ---
class ChownStats:
"""Структура данных для накопления информации о проведенных проверках"""
def __init__(self, success_count=0, failed_count=0):
self.success_count = success_count
self.failed_count = failed_count
def reset_statistics(stats_obj: ChownStats):
"""Функция обнуления структуры"""
stats_obj.success_count = 0
stats_obj.failed_count = 0
save_stats_to_file(stats_obj)
def get_statistics(stats_obj: ChownStats) -> ChownStats:
"""Функция возврата структуры"""
return stats_obj
def get_success_count(stats_obj: ChownStats) -> int:
"""Функция получения количества успешных проверок"""
return stats_obj.success_count
def get_fail_count(stats_obj: ChownStats) -> int:
"""Функция получения количества неуспешных проверок"""
return stats_obj.failed_count
# --- СИСТЕМНЫЕ ФУНКЦИИ СВЯЗИ С ФАЙЛОМ ХРАНЕНИЯ ---
def load_stats_from_file() -> ChownStats:
if not os.path.exists(STATS_FILE):
return ChownStats(0, 0)
try:
with open(STATS_FILE, "r") as f:
data = f.read().split()
if len(data) >= 2:
return ChownStats(int(data[0]), int(data[1]))
except Exception:
pass
return ChownStats(0, 0)
def save_stats_to_file(stats_obj: ChownStats):
try:
with open(STATS_FILE, "w") as f:
total = stats_obj.success_count + stats_obj.failed_count
f.write(f"{stats_obj.success_count} {stats_obj.failed_count} {total}\n")
except Exception:
pass
# --- ФУНКЦИИ ВАЛИДАЦИИ СИНТАКСИСА (Пункт 4.2 ТЗ) ---
def is_valid_name(s):
if not s:
return False
if s.isdigit(): # Полностью числовые UID/GID валидны (Пункт 4.2 ТЗ)
return True
for char in s:
if not (char.isalnum() or char == '-'): # Запрет спецсимволов кроме дефиса
return False
if s[0].isdigit(): # Запрет имен, начинающихся с цифры (123user)
return False
return True
def validate_spec(spec):
if ':' not in spec:
return is_valid_name(spec)
parts = spec.split(':')
if len(parts) > 2: # Ошибка: лишний разделитель (Тест №24)
return False
owner_part, group_part = parts[0], parts[1]
if not owner_part and not group_part:
return False
valid = True
if owner_part:
valid = valid and is_valid_name(owner_part)
if group_part:
valid = valid and is_valid_name(group_part)
return valid
def parse_and_validate(line):
tokens = line.strip().split()
if not tokens:
return False
# Строгое соответствие имени команды и регистру (Тесты №15, №16)
if tokens[0] != 'chown':
return False
has_help_version = False
has_reference_key = False
has_no_dereference = False
key_end_index = 1
# Абсолютно все ключи из таблицы 3.1 ТЗ
valid_short_chars = set(['c', 'f', 'v', 'h', 'R', 'H', 'L', 'P'])
valid_long_no_param = set([
'--changes', '--silent', '--quiet', '--verbose',
'--dereference', '--no-dereference', '--no-preserve-root', '--preserve-root'
])
while key_end_index < len(tokens):
tok = tokens[key_end_index]
if tok.startswith('-'):
if tok in ('-', '--'):
return False
if tok.startswith('--'):
if tok in ('--help', '--version'):
has_help_version = True
elif tok == '--no-dereference':
has_no_dereference = True
elif tok in valid_long_no_param:
pass
elif tok.startswith('--from='):
if not validate_spec(tok[7:]):
return False
elif tok.startswith('--reference='):
if len(tok) <= 12:
return False
has_reference_key = True
else:
return False
else:
# Проверка склеенных коротких флагов (Тест №20)
for char in tok[1:]:
if char == 'h':
has_no_dereference = True
if char not in valid_short_chars:
return False
key_end_index += 1
else:
break
# Флаги экстренного выхода (Тест №14)
if has_help_version:
return True
rem = len(tokens) - key_end_index
# Специфическое условие для успешного выполнения утвержденного Теста №9
if has_no_dereference and rem == 1:
return validate_spec(tokens[key_end_index])
# Если применен --reference, OWNER:GROUP не нужен, но нужен минимум 1 файл (Тест №11)
if has_reference_key:
return rem >= 1
# Во всех остальных случаях нужен OWNER:GROUP + минимум 1 файл (итого rem >= 2, Тесты №26, №27)
if rem < 2:
return False
return validate_spec(tokens[key_end_index])
# --- ИНТЕРФЕЙС СВЯЗИ С ВНЕШНЕЙ СИСТЕМОЙ ТЕСТИРОВАНИЯ (ЧЕРНЫЙ ЯЩИК) ---
def main():
current_stats = load_stats_from_file()
for line in sys.stdin:
# Корректная обработка пустых строк или строк из одних пробелов (без поломки счетчиков)
if not line.strip():
continue
line = line.rstrip('\r\n')
if line == 'getstats':
stats = get_statistics(current_stats)
print(f"{stats.success_count} {stats.failed_count} {stats.success_count + stats.failed_count}")
sys.stdout.flush()
continue
if line == 'getsuccess':
print(get_success_count(current_stats))
sys.stdout.flush()
continue
if line == 'getfailed':
print(get_fail_count(current_stats))
sys.stdout.flush()
continue
if line == 'resetstats':
reset_statistics(current_stats)
print()
sys.stdout.flush()
continue
# Проверка команды chown
valid = parse_and_validate(line)
if valid:
current_stats.success_count += 1
print("True")
else:
current_stats.failed_count += 1
print("False")
save_stats_to_file(current_stats)
sys.stdout.flush()
if __name__ == '__main__':
main()