Добавлены новые команды:
- list - получение списка бэкапов (снимков состояния) - restore - восстановить состояние файлов указанного снимка Изменено: - create - создание снимков выполняется через ответвление и мердж ответвленной ветки Текущие изменения позволяют создавать репозиторий, делать снимки состояния файлов и восстанавливать конкретное состояние
This commit is contained in:
		
							parent
							
								
									cdd2ec5875
								
							
						
					
					
						commit
						a193f53871
					
				
					 1 changed files with 177 additions and 3 deletions
				
			
		| 
						 | 
					@ -7,6 +7,7 @@ import std.array;
 | 
				
			||||||
import std.process;
 | 
					import std.process;
 | 
				
			||||||
import std.algorithm;
 | 
					import std.algorithm;
 | 
				
			||||||
import std.string;
 | 
					import std.string;
 | 
				
			||||||
 | 
					import std.regex;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import snag.core.exception;
 | 
					import snag.core.exception;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -14,6 +15,11 @@ class Snag {
 | 
				
			||||||
	private string[] _baseCommand;
 | 
						private string[] _baseCommand;
 | 
				
			||||||
	private SnagConfig _config;
 | 
						private SnagConfig _config;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						private bool isValidHash(string hash) {
 | 
				
			||||||
 | 
							auto hashPattern = ctRegex!r"^[a-fA-F0-9]{7}$";
 | 
				
			||||||
 | 
							return !matchFirst(hash, hashPattern).empty;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	this(SnagConfig config) {
 | 
						this(SnagConfig config) {
 | 
				
			||||||
		_baseCommand = format(
 | 
							_baseCommand = format(
 | 
				
			||||||
			"git --git-dir=%s --work-tree=%s",
 | 
								"git --git-dir=%s --work-tree=%s",
 | 
				
			||||||
| 
						 | 
					@ -67,9 +73,9 @@ class Snag {
 | 
				
			||||||
		writeln("The following list of files requires backup:");
 | 
							writeln("The following list of files requires backup:");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		/**
 | 
							/**
 | 
				
			||||||
		Цепочка выполняет разбивку по переводу на новую строку,
 | 
							* Цепочка выполняет разбивку по переводу на новую строку,
 | 
				
			||||||
		отсеивает пустые строки, перебирает в цикле имеющиеся строки,
 | 
							* отсеивает пустые строки, перебирает в цикле имеющиеся строки,
 | 
				
			||||||
		выводит только вторую часть строки (разделенную по пробелу)
 | 
							* выводит только вторую часть строки (разделенную по пробелу)
 | 
				
			||||||
		*/
 | 
							*/
 | 
				
			||||||
		result.output.split('\n').filter!(e => !e.strip.empty).each!((e) {
 | 
							result.output.split('\n').filter!(e => !e.strip.empty).each!((e) {
 | 
				
			||||||
			writefln("\t/%s", e.strip.split[1]);
 | 
								writefln("\t/%s", e.strip.split[1]);
 | 
				
			||||||
| 
						 | 
					@ -77,7 +83,45 @@ class Snag {
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	void create() {
 | 
						void create() {
 | 
				
			||||||
 | 
							/**
 | 
				
			||||||
 | 
							* Проверка файлов на состояние изменения
 | 
				
			||||||
 | 
							*/
 | 
				
			||||||
		auto result = execute(
 | 
							auto result = execute(
 | 
				
			||||||
 | 
								_baseCommand ~ ["status", "--porcelain"]
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
							if (result.status)
 | 
				
			||||||
 | 
								throw new SnagException(
 | 
				
			||||||
 | 
									"An error occurred while checking the file tracking status:\n"
 | 
				
			||||||
 | 
									~ result.output
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
							if (!result.output.length)
 | 
				
			||||||
 | 
								throw new SnagException(
 | 
				
			||||||
 | 
									"Current file state doesn't need to be archived again"
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							result = execute(
 | 
				
			||||||
 | 
								_baseCommand ~ [
 | 
				
			||||||
 | 
									"rev-parse", "--short", "HEAD"
 | 
				
			||||||
 | 
								]
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
							if (result.status)
 | 
				
			||||||
 | 
								throw new SnagException(
 | 
				
			||||||
 | 
									"Failed to retrieve current snapshot information:\n"
 | 
				
			||||||
 | 
									~ result.output
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							string currentSnapshot = result.output.strip('\n');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							result = execute(
 | 
				
			||||||
 | 
								_baseCommand ~ [
 | 
				
			||||||
 | 
									"checkout",
 | 
				
			||||||
 | 
									"-b",
 | 
				
			||||||
 | 
									"mod",
 | 
				
			||||||
 | 
									currentSnapshot
 | 
				
			||||||
 | 
								]
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							result = execute(
 | 
				
			||||||
			_baseCommand ~ ["add", "."]
 | 
								_baseCommand ~ ["add", "."]
 | 
				
			||||||
		);
 | 
							);
 | 
				
			||||||
		if (result.status)
 | 
							if (result.status)
 | 
				
			||||||
| 
						 | 
					@ -95,6 +139,136 @@ class Snag {
 | 
				
			||||||
				~ result.output
 | 
									~ result.output
 | 
				
			||||||
			);
 | 
								);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							result = execute(
 | 
				
			||||||
 | 
								_baseCommand ~ ["checkout", "master"]
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
							if (result.status)
 | 
				
			||||||
 | 
								throw new SnagException(
 | 
				
			||||||
 | 
									"Failed to perform intermediate state switching:\n"
 | 
				
			||||||
 | 
									~ result.output
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							result = execute(
 | 
				
			||||||
 | 
								_baseCommand ~ [
 | 
				
			||||||
 | 
									"merge",
 | 
				
			||||||
 | 
									"-X",
 | 
				
			||||||
 | 
									"theirs",
 | 
				
			||||||
 | 
									"--squash",
 | 
				
			||||||
 | 
									"mod"
 | 
				
			||||||
 | 
								]
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
							if (result.status)
 | 
				
			||||||
 | 
								throw new SnagException(
 | 
				
			||||||
 | 
									"Failed to retrieve changes from the new intermediate snapshot:\n"
 | 
				
			||||||
 | 
									~ result.output
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							result = execute(
 | 
				
			||||||
 | 
								_baseCommand ~ ["commit", "-m", "test"]
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
							if (result.status)
 | 
				
			||||||
 | 
								throw new SnagException(
 | 
				
			||||||
 | 
									"Failed to create backup after acquiring new intermediate snapshot:\n"
 | 
				
			||||||
 | 
									~ result.output
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
							
 | 
				
			||||||
 | 
							result = execute(
 | 
				
			||||||
 | 
								_baseCommand ~ ["branch", "-D", "mod"]
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
							if (result.status)
 | 
				
			||||||
 | 
								throw new SnagException(
 | 
				
			||||||
 | 
									"Error while deleting temporary intermediate snapshot:\n"
 | 
				
			||||||
 | 
									~ result.output
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		writeln("Backup was created successfully");
 | 
							writeln("Backup was created successfully");
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						void list() {
 | 
				
			||||||
 | 
							auto result = execute(
 | 
				
			||||||
 | 
								_baseCommand ~ [
 | 
				
			||||||
 | 
									"rev-parse", "--short", "HEAD"
 | 
				
			||||||
 | 
								]
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
							if (result.status)
 | 
				
			||||||
 | 
								throw new SnagException(
 | 
				
			||||||
 | 
									"Failed to retrieve current snapshot information:\n"
 | 
				
			||||||
 | 
									~ result.output
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							string current = result.output.strip('\n');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							result = execute(
 | 
				
			||||||
 | 
								_baseCommand ~ [
 | 
				
			||||||
 | 
									"log",
 | 
				
			||||||
 | 
									"--all",
 | 
				
			||||||
 | 
									"--date=format:%Y.%m.%d %H:%M",
 | 
				
			||||||
 | 
									"--pretty=format:%ad\t%h"
 | 
				
			||||||
 | 
								]
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
							if (result.status)
 | 
				
			||||||
 | 
								throw new SnagException(
 | 
				
			||||||
 | 
									"Failed to retrieve the list of snapshots:\n"
 | 
				
			||||||
 | 
									~ result.output
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							result.output.split('\n').each!((e) {
 | 
				
			||||||
 | 
								auto eArray = e.split('\t');
 | 
				
			||||||
 | 
								if (current == eArray[1])
 | 
				
			||||||
 | 
									writefln("    >\t%s\t%s", eArray[0], eArray[1]);
 | 
				
			||||||
 | 
								else
 | 
				
			||||||
 | 
									writefln("\t%s\t%s", eArray[0], eArray[1]);
 | 
				
			||||||
 | 
							});
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						void restore(string hash) {
 | 
				
			||||||
 | 
							if (!isValidHash(hash))
 | 
				
			||||||
 | 
								throw new SnagException(
 | 
				
			||||||
 | 
									"Invalid snapshot hash provided"
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							auto result = execute(
 | 
				
			||||||
 | 
								_baseCommand ~ ["status", "--porcelain"]
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
							if (result.status)
 | 
				
			||||||
 | 
								throw new SnagException(
 | 
				
			||||||
 | 
									"An error occurred while checking the file tracking status:\n"
 | 
				
			||||||
 | 
									~ result.output
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (result.output.length) {
 | 
				
			||||||
 | 
								result = execute(
 | 
				
			||||||
 | 
									_baseCommand ~ ["restore", "."]
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
								if (result.status)
 | 
				
			||||||
 | 
									throw new SnagException(
 | 
				
			||||||
 | 
										"Failed to reset file changes state:\n"
 | 
				
			||||||
 | 
										~ result.output
 | 
				
			||||||
 | 
									);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							result = execute(
 | 
				
			||||||
 | 
								_baseCommand ~ [
 | 
				
			||||||
 | 
									"rev-parse", hash
 | 
				
			||||||
 | 
								]
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
							if (result.status)
 | 
				
			||||||
 | 
								throw new SnagException(
 | 
				
			||||||
 | 
									"This snapshot is not available in the archive:\n"
 | 
				
			||||||
 | 
									~ result.output.split('\n')[0]
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							result = execute(
 | 
				
			||||||
 | 
								_baseCommand ~ [
 | 
				
			||||||
 | 
									"checkout", hash
 | 
				
			||||||
 | 
								]
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
							if (result.status)
 | 
				
			||||||
 | 
								throw new SnagException(
 | 
				
			||||||
 | 
									"Failed to restore the snapshot state %s:\n"
 | 
				
			||||||
 | 
									.format(hash, result.output)
 | 
				
			||||||
 | 
								);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							writeln("Backup was restored successfully");
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue