added base client-server files

This commit is contained in:
Alexander Karpov 2024-02-07 02:11:17 +03:00
commit d9f51a1a21
22 changed files with 686 additions and 0 deletions

29
.gitignore vendored Normal file
View File

@ -0,0 +1,29 @@
### IntelliJ IDEA ###
out/
!**/src/main/**/out/
!**/src/test/**/out/
### Eclipse ###
.apt_generated
.classpath
.factorypath
.project
.settings
.springBeans
.sts4-cache
bin/
!**/src/main/**/bin/
!**/src/test/**/bin/
### NetBeans ###
/nbproject/private/
/nbbuild/
/dist/
/nbdist/
/.nb-gradle/
### VS Code ###
.vscode/
### Mac OS ###
.DS_Store

6
.idea/misc.xml Normal file
View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="ProjectRootManager" version="2" languageLevel="JDK_17" default="true" project-jdk-name="17" project-jdk-type="JavaSDK">
<output url="file://$PROJECT_DIR$/out" />
</component>
</project>

8
.idea/modules.xml Normal file
View File

@ -0,0 +1,8 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="ProjectModuleManager">
<modules>
<module fileurl="file://$PROJECT_DIR$/itmo-prog-lab-5.iml" filepath="$PROJECT_DIR$/itmo-prog-lab-5.iml" />
</modules>
</component>
</project>

6
.idea/vcs.xml Normal file
View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="VcsDirectoryMappings">
<mapping directory="$PROJECT_DIR$" vcs="Git" />
</component>
</project>

100
.idea/workspace.xml Normal file
View File

@ -0,0 +1,100 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="AutoImportSettings">
<option name="autoReloadType" value="SELECTIVE" />
</component>
<component name="ChangeListManager">
<list default="true" id="4f3a5547-ba17-42ad-9cfe-ccdc381c7dc5" name="Changes" comment="">
<change afterPath="$PROJECT_DIR$/.gitignore" afterDir="false" />
<change afterPath="$PROJECT_DIR$/.idea/misc.xml" afterDir="false" />
<change afterPath="$PROJECT_DIR$/.idea/modules.xml" afterDir="false" />
<change afterPath="$PROJECT_DIR$/.idea/vcs.xml" afterDir="false" />
<change afterPath="$PROJECT_DIR$/.idea/workspace.xml" afterDir="false" />
<change afterPath="$PROJECT_DIR$/itmo-prog-lab-5.iml" afterDir="false" />
<change afterPath="$PROJECT_DIR$/src/Main.java" afterDir="false" />
</list>
<option name="SHOW_DIALOG" value="false" />
<option name="HIGHLIGHT_CONFLICTS" value="true" />
<option name="HIGHLIGHT_NON_ACTIVE_CHANGELIST" value="false" />
<option name="LAST_RESOLUTION" value="IGNORE" />
</component>
<component name="FileTemplateManagerImpl">
<option name="RECENT_TEMPLATES">
<list>
<option value="Enum" />
<option value="Class" />
</list>
</option>
</component>
<component name="Git.Settings">
<option name="RECENT_GIT_ROOT_PATH" value="$PROJECT_DIR$" />
</component>
<component name="MarkdownSettingsMigration">
<option name="stateVersion" value="1" />
</component>
<component name="ProjectColorInfo"><![CDATA[{
"associatedIndex": 2
}]]></component>
<component name="ProjectId" id="2c0VemoZcpyCQjJpTj6tw8sHgOH" />
<component name="ProjectLevelVcsManager" settingsEditedManually="true" />
<component name="ProjectViewState">
<option name="hideEmptyMiddlePackages" value="true" />
<option name="showLibraryContents" value="true" />
</component>
<component name="PropertiesComponent"><![CDATA[{
"keyToString": {
"Application.Main.executor": "Run",
"RunOnceActivity.OpenProjectViewOnStart": "true",
"RunOnceActivity.ShowReadmeOnStart": "true",
"git-widget-placeholder": "master",
"kotlin-language-version-configured": "true",
"node.js.detected.package.eslint": "true",
"node.js.detected.package.tslint": "true",
"node.js.selected.package.eslint": "(autodetect)",
"node.js.selected.package.tslint": "(autodetect)",
"nodejs_package_manager_path": "npm",
"project.structure.last.edited": "Project",
"project.structure.proportion": "0.0",
"project.structure.side.proportion": "0.0",
"settings.editor.selected.configurable": "preferences.lookFeel",
"vue.rearranger.settings.migration": "true"
}
}]]></component>
<component name="RunManager">
<configuration name="Main" type="Application" factoryName="Application" temporary="true" nameIsGenerated="true">
<option name="MAIN_CLASS_NAME" value="Main" />
<module name="itmo-prog-lab-5" />
<method v="2">
<option name="Make" enabled="true" />
</method>
</configuration>
<recent_temporary>
<list>
<item itemvalue="Application.Main" />
</list>
</recent_temporary>
</component>
<component name="SharedIndexes">
<attachedChunks>
<set>
<option value="jdk-17.0.10-corretto-17.0.10-3183f394aec4-26c513f3" />
<option value="jdk-21.0.1-openjdk-21.0.1-3183f394aec4-920bfa72" />
</set>
</attachedChunks>
</component>
<component name="SpellCheckerSettings" RuntimeDictionaries="0" Folders="0" CustomDictionaries="0" DefaultDictionary="application-level" UseSingleDictionary="true" transferred="true" />
<component name="TaskManager">
<task active="true" id="Default" summary="Default task">
<changelist id="4f3a5547-ba17-42ad-9cfe-ccdc381c7dc5" name="Changes" comment="" />
<created>1707248583469</created>
<option name="number" value="Default" />
<option name="presentableId" value="Default" />
<updated>1707248583469</updated>
<workItem from="1707248584490" duration="9894000" />
</task>
<servers />
</component>
<component name="TypeScriptGeneratedFilesManager">
<option name="version" value="3" />
</component>
</project>

84
README.md Normal file
View File

@ -0,0 +1,84 @@
# Лабраторная 5
## Текст задания
Реализовать консольное приложение, которое реализует управление коллекцией объектов в интерактивном режиме. В коллекции необходимо хранить объекты класса Ticket, описание которого приведено ниже.
Разработанная программа должна удовлетворять следующим требованиям:
Класс, коллекцией экземпляров которого управляет программа, должен реализовывать сортировку по умолчанию.
Все требования к полям класса (указанные в виде комментариев) должны быть выполнены.
Для хранения необходимо использовать коллекцию типа java.util.ArrayDequeue
При запуске приложения коллекция должна автоматически заполняться значениями из файла.
Имя файла должно передаваться программе с помощью: аргумент командной строки.
Данные должны храниться в файле в формате xml
Чтение данных из файла необходимо реализовать с помощью класса java.util.Scanner
Запись данных в файл необходимо реализовать с помощью класса java.io.OutputStreamWriter
Все классы в программе должны быть задокументированы в формате javadoc.
Программа должна корректно работать с неправильными данными (ошибки пользовательского ввода, отсутсвие прав доступа к файлу и т.п.).
В интерактивном режиме программа должна поддерживать выполнение следующих команд:
help : вывести справку по доступным командам
info : вывести в стандартный поток вывода информацию о коллекции (тип, дата инициализации, количество элементов и т.д.)
show : вывести в стандартный поток вывода все элементы коллекции в строковом представлении
add {element} : добавить новый элемент в коллекцию
update id {element} : обновить значение элемента коллекции, id которого равен заданному
remove_by_id id : удалить элемент из коллекции по его id
clear : очистить коллекцию
save : сохранить коллекцию в файл
execute_script file_name : считать и исполнить скрипт из указанного файла. В скрипте содержатся команды в таком же виде, в котором их вводит пользователь в интерактивном режиме.
exit : завершить программу (без сохранения в файл)
remove_head : вывести первый элемент коллекции и удалить его
add_if_min {element} : добавить новый элемент в коллекцию, если его значение меньше, чем у наименьшего элемента этой коллекции
remove_greater {element} : удалить из коллекции все элементы, превышающие заданный
group_counting_by_type : сгруппировать элементы коллекции по значению поля type, вывести количество элементов в каждой группе
filter_by_event event : вывести элементы, значение поля event которых равно заданному
filter_less_than_type type : вывести элементы, значение поля type которых меньше заданного
Формат ввода команд:
Все аргументы команды, являющиеся стандартными типами данных (примитивные типы, классы-оболочки, String, классы для хранения дат), должны вводиться в той же строке, что и имя команды.
Все составные типы данных (объекты классов, хранящиеся в коллекции) должны вводиться по одному полю в строку.
При вводе составных типов данных пользователю должно показываться приглашение к вводу, содержащее имя поля (например, "Введите дату рождения:")
Если поле является enum'ом, то вводится имя одной из его констант (при этом список констант должен быть предварительно выведен).
При некорректном пользовательском вводе (введена строка, не являющаяся именем константы в enum'е; введена строка вместо числа; введённое число не входит в указанные границы и т.п.) должно быть показано сообщение об ошибке и предложено повторить ввод поля.
Для ввода значений null использовать пустую строку.
Поля с комментарием "Значение этого поля должно генерироваться автоматически" не должны вводиться пользователем вручную при добавлении.
Описание хранимых в коллекции классов:
```java
public class Ticket {
private long id; //Значение поля должно быть больше 0, Значение этого поля должно быть уникальным, Значение этого поля должно генерироваться автоматически
private String name; //Поле не может быть null, Строка не может быть пустой
private Coordinates coordinates; //Поле не может быть null
private java.time.LocalDate creationDate; //Поле не может быть null, Значение этого поля должно генерироваться автоматически
private Long price; //Поле может быть null, Значение поля должно быть больше 0
private Long discount; //Поле может быть null, Значение поля должно быть больше 0, Максимальное значение поля: 100
private TicketType type; //Поле может быть null
private Event event; //Поле может быть null
}
public class Coordinates {
private int x; //Максимальное значение поля: 794
private int y;
}
public class Event {
private int id; //Значение поля должно быть больше 0, Значение этого поля должно быть уникальным, Значение этого поля должно генерироваться автоматически
private String name; //Поле не может быть null, Строка не может быть пустой
private long ticketsCount; //Значение поля должно быть больше 0
private String description; //Поле может быть null
private EventType eventType; //Поле может быть null
}
public enum TicketType {
VIP,
USUAL,
BUDGETARY,
CHEAP;
}
public enum EventType {
E_SPORTS,
FOOTBALL,
BASEBALL,
EXPOSITION;
}
```

11
itmo-prog-lab-5.iml Normal file
View File

@ -0,0 +1,11 @@
<?xml version="1.0" encoding="UTF-8"?>
<module type="JAVA_MODULE" version="4">
<component name="NewModuleRootManager" inherit-compiler-output="true">
<exclude-output />
<content url="file://$MODULE_DIR$">
<sourceFolder url="file://$MODULE_DIR$/src" isTestSource="false" />
</content>
<orderEntry type="inheritedJdk" />
<orderEntry type="sourceFolder" forTests="false" />
</component>
</module>

10
src/Main.java Normal file
View File

@ -0,0 +1,10 @@
import client.ClientApp;
import server.ServerApp;
public class Main {
public static void main(String[] args) {
ServerApp serverApp = new ServerApp();
ClientApp clientApp = new ClientApp(serverApp);
clientApp.start();
}
}

46
src/client/ClientApp.java Normal file
View File

@ -0,0 +1,46 @@
package client;
import server.ServerApp;
import java.util.Scanner;
public class ClientApp {
private ServerApp serverApp;
private Scanner scanner;
public ClientApp(ServerApp serverApp) {
this.serverApp = serverApp;
this.scanner = new Scanner(System.in);
}
public void start() {
System.out.println("Клиент запущен. Введите 'exit' для выхода.");
while (true) {
System.out.print("Введите команду: ");
if (!scanner.hasNextLine()) { // Проверка на EOF
System.out.println("Клиент завершает работу из-за отсутствия входных данных.");
break;
}
String input = scanner.nextLine();
if ("exit".equalsIgnoreCase(input)) {
System.out.println("Клиент завершает работу.");
break;
}
String[] parts = input.split(" ", 2);
String commandName = parts[0];
String[] args = parts.length > 1 ? new String[]{parts[1]} : new String[]{};
String response = sendCommand(commandName, args);
System.out.println(response);
}
}
public String sendCommand(String commandName, String[] args) {
return serverApp.executeCommand(commandName, args);
}
public static void main(String[] args) {
ServerApp serverApp = new ServerApp();
ClientApp clientApp = new ClientApp(serverApp);
clientApp.start();
}
}

View File

@ -0,0 +1,26 @@
package client;
import java.util.Scanner;
public class ConsoleManager {
private final ClientApp clientApp;
private final Scanner scanner;
public ConsoleManager(ClientApp clientApp) {
this.clientApp = clientApp;
this.scanner = new Scanner(System.in);
}
public void run() {
while (true) {
System.out.print("Введите команду: ");
String input = scanner.nextLine();
if ("exit".equalsIgnoreCase(input)) break;
String[] parts = input.split(" ", 2);
String commandName = parts[0];
String[] args = parts.length > 1 ? new String[]{parts[1]} : new String[]{};
String response = clientApp.sendCommand(commandName, args);
System.out.println(response);
}
}
}

View File

@ -0,0 +1,8 @@
package common.enums;
public enum TicketType {
VIP,
USUAL,
BUDGETARY,
CHEAP;
}

View File

@ -0,0 +1,29 @@
package common.models;
public class Coordinates {
private double x;
private float y;
// Конструктор
public Coordinates(double x, float y) {
this.x = x;
this.y = y;
}
// Геттеры и сеттеры
public double getX() {
return x;
}
public void setX(double x) {
this.x = x;
}
public float getY() {
return y;
}
public void setY(float y) {
this.y = y;
}
}

View File

@ -0,0 +1,37 @@
package common.models;
public class Event {
private long id;
private String name;
private int ticketCount;
public Event(long id, String name, int ticketCount) {
this.id = id;
this.name = name;
this.ticketCount = ticketCount;
}
public long getId() {
return id;
}
public void setId(long id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getTicketCount() {
return ticketCount;
}
public void setTicketCount(int ticketCount) {
this.ticketCount = ticketCount;
}
}

View File

@ -0,0 +1,90 @@
package common.models;
import common.enums.TicketType;
import java.time.LocalDate;
public class Ticket implements Comparable<Ticket> {
private static long nextId = 1;
private final long id;
private String name;
private Coordinates coordinates;
private final LocalDate creationDate;
private Long price;
private Long discount;
private TicketType type;
private Event event;
public Ticket(String name, Coordinates coordinates, Long price, Long discount, TicketType type, Event event) {
this.id = nextId++;
this.name = name;
this.coordinates = coordinates;
this.creationDate = LocalDate.now();
this.price = price;
this.discount = discount;
this.type = type;
this.event = event;
}
@Override
public int compareTo(Ticket o) {
return Long.compare(this.id, o.id);
}
// Геттеры и сеттеры
public long getId() {
return id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Coordinates getCoordinates() {
return coordinates;
}
public void setCoordinates(Coordinates coordinates) {
this.coordinates = coordinates;
}
public LocalDate getCreationDate() {
return creationDate;
}
public Long getPrice() {
return price;
}
public void setPrice(Long price) {
this.price = price;
}
public Long getDiscount() {
return discount;
}
public void setDiscount(Long discount) {
this.discount = discount;
}
public TicketType getType() {
return type;
}
public void setType(TicketType type) {
this.type = type;
}
public Event getEvent() {
return event;
}
public void setEvent(Event event) {
this.event = event;
}
}

View File

@ -0,0 +1,4 @@
package common.utils;
public class TicketComparator {
}

View File

@ -0,0 +1,37 @@
package server;
import server.commands.Command;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
public class CollectionManager {
private static final String CLASS_PATH = "server.commands.%sCommand";
private static final String VALIDATION_ERROR_MESSAGE = "Ошибка валидации аргументов команды.";
private static final String EXECUTION_ERROR_MESSAGE = "Ошибка выполнения команды: ";
public String executeCommand(String commandName, String[] args) {
try {
Command command = createCommand(commandName, args);
if (command.validate()) {
command.call();
return command.getResult();
} else {
return VALIDATION_ERROR_MESSAGE;
}
} catch (Exception e) {
return EXECUTION_ERROR_MESSAGE + e.getMessage();
}
}
private Command createCommand(String commandName, String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
Class<?> commandClass = Class.forName(String.format(CLASS_PATH, commandName));
Constructor<?> constructor = commandClass.getConstructor(String[].class, CollectionManager.class);
return (Command) constructor.newInstance((Object) args, this);
}
public String show() {
// Your logic to display the collection
return "Элементы коллекции: ...";
}
}

View File

@ -0,0 +1,35 @@
package server;
import server.commands.Command;
public class CommandProcessor {
public static String process(String commandLine, CollectionManager collectionManager) {
String[] parts = commandLine.trim().split(" ", 2);
String commandName = parts[0].toLowerCase();
String[] args = parts.length > 1 ? new String[]{parts[1]} : new String[]{};
try {
String className = "server.commands." + Character.toUpperCase(commandName.charAt(0)) + commandName.substring(1) + "Command";
Class<?> clazz = Class.forName(className);
// Создание экземпляра команды с CollectionManager и args
Command command = (Command) clazz.getConstructor(CollectionManager.class, String[].class).newInstance(collectionManager, (Object)args);
if (command.requiresArguments() && args.length == 0) {
return "Команда требует аргументы.";
}
if (command.validate()) {
command.call();
return command.getResult();
} else {
return "Ошибка валидации аргументов команды.";
}
} catch (ClassNotFoundException e) {
return "Команда не найдена.";
} catch (Exception e) {
e.printStackTrace();
return "Ошибка выполнения команды: " + e.getMessage();
}
}
}

View File

@ -0,0 +1,36 @@
package server;
import common.models.Ticket;
import java.io.*;
import java.nio.charset.StandardCharsets;
import java.util.ArrayDeque;
import java.util.Scanner;
public class FileManager {
private final String fileName;
public FileManager(String fileName) {
this.fileName = fileName;
}
public ArrayDeque<Ticket> loadCollection() {
ArrayDeque<Ticket> tickets = new ArrayDeque<>();
try (Scanner scanner = new Scanner(new File(fileName))) {
// Чтение и парсинг XML файла для заполнения коллекции tickets
// Примерный код, требуется реализация парсинга XML
} catch (FileNotFoundException e) {
System.err.println("Файл не найден: " + e.getMessage());
}
return tickets;
}
public void saveCollection(ArrayDeque<Ticket> tickets) {
try (OutputStreamWriter writer = new OutputStreamWriter(new FileOutputStream(fileName), StandardCharsets.UTF_8)) {
// Запись коллекции tickets в файл в формате XML
// Примерный код, требуется реализация форматирования и записи XML
} catch (IOException e) {
System.err.println("Ошибка при записи файла: " + e.getMessage());
}
}
}

10
src/server/ServerApp.java Normal file
View File

@ -0,0 +1,10 @@
package server;
public class ServerApp {
private final CollectionManager collectionManager = new CollectionManager();
public String executeCommand(String command, String[] args) {
// Обработка команды и возвращение результата выполнения
return CommandProcessor.process(command, collectionManager);
}
}

View File

@ -0,0 +1,21 @@
package server.commands;
public abstract class Command {
protected String[] args;
protected String result;
public Command(String[] args) {
this.args = args;
}
public abstract boolean validate();
public abstract void call();
public abstract boolean requiresArguments();
public String getResult() {
return result;
}
public void setArgs(String[] args) {
}
}

View File

@ -0,0 +1,27 @@
package server.commands;
import server.CollectionManager;
public class HelpCommand extends Command {
private CollectionManager collectionManager;
public HelpCommand(CollectionManager collectionManager, String[] args) {
super(args);
this.collectionManager = collectionManager;
}
@Override
public boolean validate() {
return args.length == 0;
}
@Override
public void call() {
result = "Список доступных команд: ...";
}
@Override
public boolean requiresArguments() {
return false;
}
}

View File

@ -0,0 +1,26 @@
package server.commands;
import server.CollectionManager;
public class ShowCommand extends Command {
private CollectionManager collectionManager;
public ShowCommand(CollectionManager collectionManager, String[] args) {
super(args);
this.collectionManager = collectionManager;
}
@Override
public boolean validate() {
return args.length == 0;
}
@Override
public void call() {
result = collectionManager.show();
}
public boolean requiresArguments() {
return false;
}
}