2016-06-23 18:31:55 +02:00
|
|
|
"use strict";
|
2017-03-04 21:10:46 +01:00
|
|
|
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
|
|
return new (P || (P = Promise))(function (resolve, reject) {
|
|
|
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
|
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
|
|
function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
|
|
|
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
|
|
});
|
|
|
|
};
|
|
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
2016-06-23 18:31:55 +02:00
|
|
|
require("typings-global");
|
2016-07-17 17:34:15 +02:00
|
|
|
const plugins = require("./smartfile.plugins");
|
|
|
|
const SmartfileInterpreter = require("./smartfile.interpreter");
|
2017-03-04 21:10:46 +01:00
|
|
|
const smartfile_classes_smartfile_1 = require("./smartfile.classes.smartfile");
|
2016-09-24 21:42:45 +02:00
|
|
|
const memory = require("./smartfile.memory");
|
2016-06-23 18:31:55 +02:00
|
|
|
/*===============================================================
|
2016-06-24 03:36:51 +02:00
|
|
|
============================ Checks =============================
|
|
|
|
===============================================================*/
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param filePath
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
exports.fileExistsSync = function (filePath) {
|
2016-07-17 17:34:15 +02:00
|
|
|
let fileExistsBool = false;
|
2016-06-24 03:36:51 +02:00
|
|
|
try {
|
2016-07-19 20:28:28 +02:00
|
|
|
plugins.fsExtra.readFileSync(filePath);
|
2016-06-24 03:36:51 +02:00
|
|
|
fileExistsBool = true;
|
|
|
|
}
|
|
|
|
catch (err) {
|
|
|
|
fileExistsBool = false;
|
|
|
|
}
|
|
|
|
return fileExistsBool;
|
|
|
|
};
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param filePath
|
|
|
|
* @returns {any}
|
|
|
|
*/
|
|
|
|
exports.fileExists = function (filePath) {
|
2016-09-17 23:11:44 +02:00
|
|
|
let done = plugins.q.defer();
|
2017-01-21 00:47:48 +01:00
|
|
|
plugins.fs.access(filePath, 4, function (err) {
|
2016-07-01 06:07:58 +02:00
|
|
|
err ? done.reject(err) : done.resolve();
|
2016-06-24 03:36:51 +02:00
|
|
|
});
|
|
|
|
return done.promise;
|
|
|
|
};
|
|
|
|
/**
|
|
|
|
* Checks if given path points to an existing directory
|
|
|
|
*/
|
|
|
|
exports.isDirectory = function (pathArg) {
|
2016-07-19 20:28:28 +02:00
|
|
|
return plugins.fsExtra.statSync(pathArg).isDirectory();
|
2016-06-24 03:36:51 +02:00
|
|
|
};
|
|
|
|
/**
|
|
|
|
* Checks if a given path points to an existing file
|
|
|
|
*/
|
|
|
|
exports.isFile = function (pathArg) {
|
2016-07-19 20:28:28 +02:00
|
|
|
return plugins.fsExtra.statSync(pathArg).isFile();
|
2016-06-24 03:36:51 +02:00
|
|
|
};
|
|
|
|
/*===============================================================
|
2016-06-23 18:31:55 +02:00
|
|
|
============================ FS ACTIONS =========================
|
|
|
|
===============================================================*/
|
2016-09-24 21:42:45 +02:00
|
|
|
/**
|
|
|
|
* copies a file from A to B on the local disk
|
|
|
|
*/
|
|
|
|
exports.copy = function (fromArg, toArg) {
|
|
|
|
let done = plugins.q.defer();
|
|
|
|
plugins.fsExtra.copy(fromArg, toArg, {}, function () {
|
|
|
|
done.resolve();
|
|
|
|
});
|
|
|
|
return done.promise;
|
|
|
|
};
|
|
|
|
/**
|
|
|
|
* copies a file SYNCHRONOUSLY from A to B on the local disk
|
|
|
|
*/
|
|
|
|
exports.copySync = function (fromArg, toArg) {
|
|
|
|
plugins.fsExtra.copySync(fromArg, toArg);
|
|
|
|
return true;
|
|
|
|
};
|
2016-06-28 06:57:51 +02:00
|
|
|
/**
|
|
|
|
* ensures that a directory is in place
|
|
|
|
*/
|
2016-07-17 17:34:15 +02:00
|
|
|
exports.ensureDir = (dirPathArg) => {
|
2016-09-17 23:11:44 +02:00
|
|
|
let done = plugins.q.defer();
|
2016-07-19 20:28:28 +02:00
|
|
|
plugins.fsExtra.ensureDir(dirPathArg, done.resolve);
|
2016-06-28 06:57:51 +02:00
|
|
|
return done.promise;
|
|
|
|
};
|
|
|
|
/**
|
|
|
|
* ensures that a directory is in place
|
|
|
|
*/
|
2016-07-17 17:34:15 +02:00
|
|
|
exports.ensureDirSync = (dirPathArg) => {
|
2016-07-19 20:28:28 +02:00
|
|
|
plugins.fsExtra.ensureDirSync(dirPathArg);
|
2016-06-28 06:57:51 +02:00
|
|
|
};
|
2017-01-01 02:45:53 +01:00
|
|
|
/**
|
|
|
|
* ensure an empty directory
|
|
|
|
* @executes ASYNC
|
|
|
|
*/
|
|
|
|
exports.ensureEmptyDir = (dirPathArg) => {
|
|
|
|
let done = plugins.q.defer();
|
|
|
|
plugins.fsExtra.ensureDir(dirPathArg, () => {
|
|
|
|
plugins.fsExtra.emptyDir(dirPathArg, done.resolve);
|
|
|
|
});
|
|
|
|
return done.promise;
|
|
|
|
};
|
|
|
|
/**
|
|
|
|
* ensure an empty directory
|
|
|
|
* @executes SYNC
|
|
|
|
*/
|
|
|
|
exports.ensureEmptyDirSync = (dirPathArg) => {
|
|
|
|
plugins.fsExtra.ensureDirSync(dirPathArg);
|
|
|
|
plugins.fsExtra.emptyDirSync(dirPathArg);
|
|
|
|
};
|
2016-06-23 18:39:02 +02:00
|
|
|
/**
|
2016-09-24 21:42:45 +02:00
|
|
|
* ensures that a file is on disk
|
|
|
|
* @param filePath the filePath to ensureDir
|
|
|
|
* @param the fileContent to place into a new file in case it doesn't exist yet
|
|
|
|
* @returns Promise<void>
|
|
|
|
* @exec ASYNC
|
2016-06-23 18:39:02 +02:00
|
|
|
*/
|
2016-09-24 21:42:45 +02:00
|
|
|
exports.ensureFile = (filePathArg, initFileStringArg) => {
|
2016-09-20 17:56:49 +02:00
|
|
|
let done = plugins.q.defer();
|
2016-09-24 21:42:45 +02:00
|
|
|
exports.ensureFileSync(filePathArg, initFileStringArg);
|
|
|
|
done.resolve();
|
2016-06-23 18:31:55 +02:00
|
|
|
return done.promise;
|
|
|
|
};
|
2016-06-23 18:39:02 +02:00
|
|
|
/**
|
2016-09-24 21:42:45 +02:00
|
|
|
* ensures that a file is on disk
|
|
|
|
* @param filePath the filePath to ensureDir
|
|
|
|
* @param the fileContent to place into a new file in case it doesn't exist yet
|
|
|
|
* @returns Promise<void>
|
|
|
|
* @exec SYNC
|
2016-06-23 18:39:02 +02:00
|
|
|
*/
|
2016-09-24 21:42:45 +02:00
|
|
|
exports.ensureFileSync = (filePathArg, initFileStringArg) => {
|
|
|
|
if (exports.fileExistsSync(filePathArg)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
memory.toFsSync(initFileStringArg, filePathArg);
|
|
|
|
}
|
2016-06-23 18:31:55 +02:00
|
|
|
};
|
2016-06-23 18:39:02 +02:00
|
|
|
/**
|
|
|
|
* removes a file or folder from local disk
|
|
|
|
*/
|
2016-06-23 18:31:55 +02:00
|
|
|
exports.remove = function (pathArg) {
|
2016-09-20 17:56:49 +02:00
|
|
|
let done = plugins.q.defer();
|
2016-07-19 20:28:28 +02:00
|
|
|
plugins.fsExtra.remove(pathArg, function () {
|
2016-06-23 18:31:55 +02:00
|
|
|
done.resolve();
|
|
|
|
});
|
|
|
|
return done.promise;
|
|
|
|
};
|
2016-06-23 18:39:02 +02:00
|
|
|
/**
|
|
|
|
* removes a file SYNCHRONOUSLY from local disk
|
|
|
|
*/
|
2016-06-23 18:31:55 +02:00
|
|
|
exports.removeSync = function (pathArg) {
|
2016-07-19 20:28:28 +02:00
|
|
|
plugins.fsExtra.removeSync(pathArg);
|
2016-06-23 18:31:55 +02:00
|
|
|
return true;
|
|
|
|
};
|
2016-09-29 14:17:46 +02:00
|
|
|
/**
|
|
|
|
* removes an array of filePaths from disk
|
|
|
|
*/
|
|
|
|
exports.removeMany = function (filePathArrayArg) {
|
|
|
|
let promiseArray = [];
|
|
|
|
for (let filePath of filePathArrayArg) {
|
|
|
|
promiseArray.push(exports.remove(filePath));
|
|
|
|
}
|
2017-01-21 00:47:48 +01:00
|
|
|
return Promise.all(promiseArray);
|
2016-09-29 14:17:46 +02:00
|
|
|
};
|
|
|
|
/**
|
|
|
|
* like removeFilePathArray but SYNCHRONOUSLY
|
|
|
|
*/
|
|
|
|
exports.removeManySync = function (filePathArrayArg) {
|
|
|
|
for (let filePath of filePathArrayArg) {
|
|
|
|
exports.removeSync(filePath);
|
|
|
|
}
|
|
|
|
};
|
2016-06-23 18:31:55 +02:00
|
|
|
/*===============================================================
|
|
|
|
============================ Write/Read =========================
|
|
|
|
===============================================================*/
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param filePathArg
|
|
|
|
* @param fileTypeArg
|
|
|
|
* @returns {any}
|
|
|
|
*/
|
|
|
|
exports.toObjectSync = function (filePathArg, fileTypeArg) {
|
2016-07-19 20:28:28 +02:00
|
|
|
let fileString = plugins.fsExtra.readFileSync(filePathArg, 'utf8');
|
2016-07-17 17:34:15 +02:00
|
|
|
let fileType;
|
2016-06-23 18:31:55 +02:00
|
|
|
fileTypeArg ? fileType = fileTypeArg : fileType = SmartfileInterpreter.filetype(filePathArg);
|
|
|
|
return SmartfileInterpreter.objectFile(fileString, fileType);
|
|
|
|
};
|
|
|
|
/**
|
|
|
|
* reads a file content to a String
|
|
|
|
* @param filePath
|
|
|
|
* @returns {string|Buffer|any}
|
|
|
|
*/
|
|
|
|
exports.toStringSync = function (filePath) {
|
2017-03-12 00:06:56 +01:00
|
|
|
let fileString = plugins.fsExtra.readFileSync(filePath, 'utf8');
|
2016-06-23 18:31:55 +02:00
|
|
|
return fileString;
|
|
|
|
};
|
2017-03-04 21:10:46 +01:00
|
|
|
exports.fileTreeToObject = (dirPathArg, miniMatchFilter) => __awaiter(this, void 0, void 0, function* () {
|
|
|
|
let fileTree = yield exports.listFileTree(dirPathArg, miniMatchFilter);
|
|
|
|
let smartfileArray = [];
|
|
|
|
for (let filePath of fileTree) {
|
|
|
|
smartfileArray.push(new smartfile_classes_smartfile_1.Smartfile({
|
|
|
|
path: filePath,
|
2017-03-15 15:25:41 +01:00
|
|
|
contentBuffer: new Buffer(exports.toStringSync(plugins.path.join(dirPathArg, filePath)))
|
2017-03-04 21:10:46 +01:00
|
|
|
}));
|
|
|
|
}
|
|
|
|
return smartfileArray;
|
|
|
|
});
|
2016-06-23 18:31:55 +02:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param filePathArg
|
|
|
|
* @param options
|
|
|
|
* @returns {number}
|
|
|
|
*/
|
2016-07-17 17:34:15 +02:00
|
|
|
exports.toVinylSync = function (filePathArg, options = {}) {
|
2016-06-23 18:31:55 +02:00
|
|
|
return plugins.vinylFile.readSync(filePathArg, options);
|
|
|
|
};
|
|
|
|
/**
|
|
|
|
* lets you reload files hot.
|
|
|
|
* @param path
|
|
|
|
* @returns {any}
|
|
|
|
*/
|
|
|
|
exports.requireReload = function (path) {
|
|
|
|
return plugins.requireReload(path);
|
|
|
|
};
|
2016-06-23 18:39:02 +02:00
|
|
|
/**
|
|
|
|
* lists Folders in a directory on local disk
|
2016-06-28 06:57:51 +02:00
|
|
|
* @returns Promise
|
2016-06-23 18:39:02 +02:00
|
|
|
*/
|
2016-06-28 08:40:22 +02:00
|
|
|
exports.listFolders = function (pathArg, regexFilter) {
|
2016-09-17 23:11:44 +02:00
|
|
|
let done = plugins.q.defer();
|
2016-07-19 20:28:28 +02:00
|
|
|
let folderArray = plugins.fsExtra.readdirSync(pathArg).filter(function (file) {
|
|
|
|
return plugins.fsExtra.statSync(plugins.path.join(pathArg, file)).isDirectory();
|
2016-06-23 18:31:55 +02:00
|
|
|
});
|
2016-06-28 08:40:22 +02:00
|
|
|
if (regexFilter) {
|
2016-07-17 17:34:15 +02:00
|
|
|
folderArray = folderArray.filter((fileItem) => {
|
2016-06-28 09:59:59 +02:00
|
|
|
return regexFilter.test(fileItem);
|
2016-06-28 08:40:22 +02:00
|
|
|
});
|
|
|
|
}
|
2016-06-23 18:31:55 +02:00
|
|
|
done.resolve(folderArray);
|
|
|
|
return done.promise;
|
|
|
|
};
|
2016-06-23 18:39:02 +02:00
|
|
|
/**
|
|
|
|
* lists Folders SYNCHRONOUSLY in a directory on local disk
|
2016-06-28 06:57:51 +02:00
|
|
|
* @returns an array with the folder names as strings
|
2016-06-23 18:39:02 +02:00
|
|
|
*/
|
2016-06-28 08:40:22 +02:00
|
|
|
exports.listFoldersSync = function (pathArg, regexFilter) {
|
2016-07-19 20:28:28 +02:00
|
|
|
let folderArray = plugins.fsExtra.readdirSync(pathArg).filter(function (file) {
|
|
|
|
return plugins.fsExtra.statSync(plugins.path.join(pathArg, file)).isDirectory();
|
2016-06-23 18:39:02 +02:00
|
|
|
});
|
2016-06-28 08:40:22 +02:00
|
|
|
if (regexFilter) {
|
2016-07-17 17:34:15 +02:00
|
|
|
folderArray = folderArray.filter((fileItem) => {
|
2016-06-28 09:59:59 +02:00
|
|
|
return regexFilter.test(fileItem);
|
2016-06-28 08:40:22 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
return folderArray;
|
|
|
|
};
|
|
|
|
/**
|
|
|
|
* lists Files in a directory on local disk
|
|
|
|
* @returns Promise
|
|
|
|
*/
|
|
|
|
exports.listFiles = function (pathArg, regexFilter) {
|
2016-09-17 23:11:44 +02:00
|
|
|
let done = plugins.q.defer();
|
2016-07-19 20:28:28 +02:00
|
|
|
let fileArray = plugins.fsExtra.readdirSync(pathArg).filter(function (file) {
|
|
|
|
return plugins.fsExtra.statSync(plugins.path.join(pathArg, file)).isFile();
|
2016-06-28 08:40:22 +02:00
|
|
|
});
|
|
|
|
if (regexFilter) {
|
2016-07-17 17:34:15 +02:00
|
|
|
fileArray = fileArray.filter((fileItem) => {
|
2016-06-28 09:59:59 +02:00
|
|
|
return regexFilter.test(fileItem);
|
2016-06-28 08:40:22 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
done.resolve(fileArray);
|
2016-06-28 09:59:59 +02:00
|
|
|
return done.promise;
|
2016-06-28 08:40:22 +02:00
|
|
|
};
|
|
|
|
/**
|
|
|
|
* lists Files SYNCHRONOUSLY in a directory on local disk
|
|
|
|
* @returns an array with the folder names as strings
|
|
|
|
*/
|
|
|
|
exports.listFilesSync = function (pathArg, regexFilter) {
|
2016-07-19 20:28:28 +02:00
|
|
|
let fileArray = plugins.fsExtra.readdirSync(pathArg).filter(function (file) {
|
|
|
|
return plugins.fsExtra.statSync(plugins.path.join(pathArg, file)).isFile();
|
2016-06-28 08:40:22 +02:00
|
|
|
});
|
|
|
|
if (regexFilter) {
|
2016-07-17 17:34:15 +02:00
|
|
|
fileArray = fileArray.filter((fileItem) => {
|
2016-06-28 09:59:59 +02:00
|
|
|
return regexFilter.test(fileItem);
|
2016-06-28 08:40:22 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
return fileArray;
|
|
|
|
};
|
|
|
|
/**
|
|
|
|
* lists all items (folders AND files) in a directory on local disk
|
2016-09-17 23:11:44 +02:00
|
|
|
* @returns Promise<string[]>
|
2016-06-28 08:40:22 +02:00
|
|
|
*/
|
|
|
|
exports.listAllItems = function (pathArg, regexFilter) {
|
2016-09-17 23:11:44 +02:00
|
|
|
let done = plugins.q.defer();
|
2016-07-19 20:28:28 +02:00
|
|
|
let allItmesArray = plugins.fsExtra.readdirSync(pathArg);
|
2016-06-28 08:40:22 +02:00
|
|
|
if (regexFilter) {
|
2016-07-17 17:34:15 +02:00
|
|
|
allItmesArray = allItmesArray.filter((fileItem) => {
|
2016-06-28 09:59:59 +02:00
|
|
|
return regexFilter.test(fileItem);
|
2016-06-28 08:40:22 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
;
|
|
|
|
done.resolve(allItmesArray);
|
2016-06-28 09:59:59 +02:00
|
|
|
return done.promise;
|
2016-06-28 08:40:22 +02:00
|
|
|
};
|
|
|
|
/**
|
2016-09-17 23:11:44 +02:00
|
|
|
* lists all items (folders AND files) in a directory on local disk
|
2016-06-28 08:40:22 +02:00
|
|
|
* @returns an array with the folder names as strings
|
2016-09-17 23:11:44 +02:00
|
|
|
* @executes SYNC
|
2016-06-28 08:40:22 +02:00
|
|
|
*/
|
|
|
|
exports.listAllItemsSync = function (pathArg, regexFilter) {
|
2016-07-19 20:28:28 +02:00
|
|
|
let allItmesArray = plugins.fsExtra.readdirSync(pathArg).filter(function (file) {
|
|
|
|
return plugins.fsExtra.statSync(plugins.path.join(pathArg, file)).isFile();
|
2016-06-28 08:40:22 +02:00
|
|
|
});
|
|
|
|
if (regexFilter) {
|
2016-07-17 17:34:15 +02:00
|
|
|
allItmesArray = allItmesArray.filter((fileItem) => {
|
2016-06-28 09:59:59 +02:00
|
|
|
return regexFilter.test(fileItem);
|
2016-06-28 08:40:22 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
return allItmesArray;
|
2016-06-23 18:39:02 +02:00
|
|
|
};
|
2016-09-17 23:11:44 +02:00
|
|
|
/**
|
|
|
|
* lists a file tree using a miniMatch filter
|
2016-09-30 16:16:11 +02:00
|
|
|
* note: if the miniMatch Filter is an absolute path, the cwdArg will be omitted
|
2016-09-17 23:11:44 +02:00
|
|
|
* @returns Promise<string[]> string array with the absolute paths of all matching files
|
|
|
|
*/
|
2016-09-30 16:16:11 +02:00
|
|
|
exports.listFileTree = (dirPathArg, miniMatchFilter) => {
|
2016-09-17 23:11:44 +02:00
|
|
|
let done = plugins.q.defer();
|
2016-09-30 16:16:11 +02:00
|
|
|
// handle absolute miniMatchFilter
|
|
|
|
let dirPath;
|
|
|
|
if (plugins.path.isAbsolute(miniMatchFilter)) {
|
|
|
|
dirPath = '/';
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dirPath = dirPathArg;
|
|
|
|
}
|
2016-07-17 17:34:15 +02:00
|
|
|
let options = {
|
2016-07-01 01:37:48 +02:00
|
|
|
cwd: dirPath
|
|
|
|
};
|
2016-07-17 17:34:15 +02:00
|
|
|
plugins.glob(miniMatchFilter, options, (err, files) => {
|
2016-07-01 01:37:48 +02:00
|
|
|
if (err) {
|
|
|
|
console.log(err);
|
2016-07-01 06:07:58 +02:00
|
|
|
done.reject(err);
|
2016-07-01 01:37:48 +02:00
|
|
|
}
|
|
|
|
done.resolve(files);
|
|
|
|
});
|
|
|
|
return done.promise;
|
|
|
|
};
|
2017-03-15 15:25:41 +01:00
|
|
|
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic21hcnRmaWxlLmZzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vdHMvc21hcnRmaWxlLmZzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7QUFBQSwwQkFBdUI7QUFFdkIsK0NBQStDO0FBQy9DLGdFQUFnRTtBQUVoRSwrRUFBeUQ7QUFFekQsNkNBQTRDO0FBQzVDOztpRUFFaUU7QUFFakU7Ozs7R0FJRztBQUNRLFFBQUEsY0FBYyxHQUFHLFVBQVUsUUFBUTtJQUM1QyxJQUFJLGNBQWMsR0FBWSxLQUFLLENBQUE7SUFDbkMsSUFBSSxDQUFDO1FBQ0gsT0FBTyxDQUFDLE9BQU8sQ0FBQyxZQUFZLENBQUMsUUFBUSxDQUFDLENBQUE7UUFDdEMsY0FBYyxHQUFHLElBQUksQ0FBQTtJQUN2QixDQUFDO0lBQUMsS0FBSyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztRQUNiLGNBQWMsR0FBRyxLQUFLLENBQUE7SUFDeEIsQ0FBQztJQUNELE1BQU0sQ0FBQyxjQUFjLENBQUE7QUFDdkIsQ0FBQyxDQUFBO0FBRUQ7Ozs7R0FJRztBQUNRLFFBQUEsVUFBVSxHQUFHLFVBQVUsUUFBUTtJQUN4QyxJQUFJLElBQUksR0FBRyxPQUFPLENBQUMsQ0FBQyxDQUFDLEtBQUssRUFBRSxDQUFBO0lBQzVCLE9BQU8sQ0FBQyxFQUFFLENBQUMsTUFBTSxDQUFDLFFBQVEsRUFBRSxDQUFDLEVBQUUsVUFBVSxHQUFHO1FBQzFDLEdBQUcsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxHQUFHLElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQTtJQUN6QyxDQUFDLENBQUMsQ0FBQTtJQUNGLE1BQU0sQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFBO0FBQ3JCLENBQUMsQ0FBQTtBQUVEOztHQUVHO0FBQ1EsUUFBQSxXQUFXLEdBQUcsVUFBVSxPQUFPO0lBQ3hDLE1BQU0sQ0FBQyxPQUFPLENBQUMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxPQUFPLENBQUMsQ0FBQyxXQUFXLEVBQUUsQ0FBQTtBQUN4RCxDQUFDLENBQUE7QUFFRDs7R0FFRztBQUNRLFFBQUEsTUFBTSxHQUFHLFVBQVUsT0FBTztJQUNuQyxNQUFNLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDLENBQUMsTUFBTSxFQUFFLENBQUE7QUFDbkQsQ0FBQyxDQUFBO0FBRUQ7O2lFQUVpRTtBQUVqRTs7R0FFRztBQUNRLFFBQUEsSUFBSSxHQUFHLFVBQVUsT0FBZSxFQUFFLEtBQWE7SUFDeEQsSUFBSSxJQUFJLEdBQUcsT0FBTyxDQUFDLENBQUMsQ0FBQyxLQUFLLEVBQUUsQ0FBQTtJQUM1QixPQUFPLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxPQUFPLEVBQUUsS0FBSyxFQUFFLEVBQUUsRUFBRTtRQUN2QyxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUE7SUFDaEIsQ0FBQyxDQUFDLENBQUE7SUFDRixNQUFNLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQTtBQUNyQixDQUFDLENBQUE7QUFFRDs7R0FFRztBQUNRLFFBQUEsUUFBUSxHQUFHLFVBQVUsT0FBZSxFQUFFLEtBQWE7SUFDNUQsT0FBTyxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUMsT0FBTyxFQUFFLEtBQUssQ0FBQyxDQUFBO0lBQ3hDLE1BQU0sQ0FBQyxJQUFJLENBQUE7QUFDYixDQUFDLENBQUE7QUFFRDs7R0FFRztBQUNRLFFBQUEsU0FBUyxHQUFHLENBQUMsVUFBa0I7SUFDeEMsSUFBSSxJQUFJLEdBQUcsT0FBTyxDQUFDLENBQUMsQ0FBQyxLQUFLLEVBQUUsQ0FBQTtJQUM1QixPQUFPLENBQUMsT0FBTyxDQUFDLFNBQVMsQ0FBQyxVQUFVLEVBQUUsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFBO0lBQ25ELE1BQU0sQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFBO0FBQ3JCLENBQUMsQ0FBQTtBQUVEOztHQUVHO0FBQ1EsUUFBQSxhQUFhLEdBQUcsQ0FBQyxVQUFrQjtJQUM1QyxPQUFPLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQyxVQUFVLENBQUMsQ0FBQTtBQUMzQyxDQUFDLENBQUE7QUFFRDs7O0dBR0c7QUFDUSxRQUFBLGNBQWMsR0FBRyxDQUFDLFVBQWtCO0lBQzdDLElBQUksSUFBSSxHQUFHLE9BQU8sQ0FBQyxDQUFDLENBQUMsS0FBSyxFQUFFLENBQUE7SUFDNUIsT0FBTyxDQUFDLE9BQU8sQ0FBQyxTQUFTLENBQUMsVUFBVSxFQUFFO1FBQ3BDLE9BQU8sQ0FBQyxPQUFPLENBQUMsUUFBUSxDQUFDLFVBQVUsRUFBRSxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUE7SUFDcEQsQ0FBQyxDQUFDLENBQUE7SUFDRixNQUFNLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQTtBQUNyQixDQUFDLENBQUE7QUFFRDs7O0dBR0c7QUFDUSxRQUFBLGtCQUFrQixHQUFHLENBQUMsVUFBa0I7SUFDakQsT0FBTyxDQUFDLE9BQU8sQ0FBQyxhQUFhLENBQUMsVUFBVSxDQUFDLENBQUE7SUFDekMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxZQUFZLENBQUMsVUFBVSxDQUFDLENBQUE7QUFDMUMsQ0FBQyxDQUFBO0FBRUQ7Ozs7OztHQU1HO0FBQ1EsUUFBQSxVQUFVLEdBQUcsQ0FBQyxXQUFXLEVBQUUsaUJBQWlCO0lBQ3JELElBQUksSUFBSSxHQUFHLE9BQU8sQ0FBQyxDQUFDLENBQUMsS0FBSyxFQUFRLENBQUE7SUFDbEMsc0JBQWMsQ0FBQyxXQUFXLEVBQUUsaUJBQWlCLENBQUMsQ0FBQTtJQUM5QyxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUE7SUFDZCxNQUFNLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQTtBQUNyQixDQUFDLENBQUE7QUFFRDs7Ozs7O0dBTUc7QUFDUSxRQUFBLGNBQWMsR0FBRyxDQUFDLFdBQW1CLEVBQUUsaUJBQXlCO0lBQ3pFLEVBQUUsQ0FBQyxDQUFDLHNCQUFjLENBQUMsV0FBVyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ2hDLE1BQU0sQ0FBQyxJQUFJLENBQUE7SUFDYixDQUFDO0lBQUMsSUFBSSxDQUFDLENBQUM7UUFDTixNQUFNLENBQUMsUUFBUSxDQUFDLGlCQUFpQixFQUFFLFdBQVcsQ0FBQyxDQUFBO0lBQ2pELENBQUM7QUFDSCxDQUFDLENBQUE7QUFFRDs7R0FFRztBQUNRLFFBQUEsTUFBTSxHQUFHLFVBQVUsT0FBZTtJQUMzQyxJQUFJLElBQUksR0FBRyxPQUFPLENBQUMsQ0FBQyxDQUFDLEtBQUssRUFBUSxDQUFBO0lBQ2xDLE9BQU8sQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLE9BQU8sRUFBRTtRQUM5QixJQUFJLENBQUMsT0FBTyxFQUFFLENBQUE7SUFDaEIsQ0FBQyxDQUFDLENBQUE7SUFDRixNQUFNLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQTtBQUNyQixDQUFDLENBQUE7Q
|