Artifact [d17d8e0f9e] Вы: nobody
Вход

Artifact d17d8e0f9ecb100a4fef3082d5fcd52c605a271f:


$engine JScript
$uname dvcs_fossil
$dname Backend к dvcs fossil
$addin global

// (c) Сосна Евгений shenja at sosna.zp.ua
// Скрипт - Backend к fossil для отображения версионного контроля. 

global.connectGlobals(SelfScript)

var pfFossilPath                    = "ExtFilesDVCS/fossilpath"
var pfFossilPathBase                = "ExtFilesDVCS/fossilpathbase"
profileRoot.createValue(pfFossilPath, ".\\fossil.exe", pflSnegopat) //по умолчанию будем указывать путь к fossil относительно корня. 
profileRoot.createValue(pfFossilPathBase, "", pflBase)

// Настройки для fossil 
var PathToFossil = "";
var мPathToFossilBase = "";
var мPathToFossil  = "";
мPathToFossilBase = profileRoot.getValue(pfFossilPathBase)
мPathToFossil = profileRoot.getValue(pfFossilPath)
// если не создаеться fso, необходимо в сделать в c:\windows\system32 
// regsvr32.exe scrrun.dll 
// после этого fso заработает. Во всем виновата корпорация добра http://social.technet.microsoft.com/Forums/ru/windowsserverru/thread/28d55900-145b-466b-93d4-74e08006c72f
var FSO = new ActiveXObject("Scripting.FileSystemObject");

if (мPathToFossilBase!=''){
    if (мPathToFossilBase.substr(0,1) == ".") {
        var мPathToFossilBase = FSO.GetAbsolutePathName(FSO.buildPath(mainFolder, мPathToFossilBase))
        }
    var f = v8New("File", мPathToFossilBase); 
    if (f.Exist()) {
        PathToFossil = мPathToFossilBase;
    }
}
if (PathToFossil=='' && мPathToFossil!='') { //прочтем настройки снегопата
    if (мPathToFossil.substr(0,1) == ".") {
        var мPathToFossil = FSO.GetAbsolutePathName(FSO.buildPath(mainFolder, мPathToFossil))
        }
    var f = v8New("File", мPathToFossil); 
    if (f.Exist()) {
        PathToFossil = мPathToFossil;
    }
}
if (PathToFossil == '') {
    мPathToFossil = "fossil.exe";
    PathToFossil = мPathToFossil;
}
if (PathToFossil.indexOf(" ")!=-1) {
    PathToFossil = '"'+PathToFossil+'"'
}


var ForReading = 1, ForWriting = 2, ForAppending = 8;
var WshShell = new ActiveXObject("WScript.Shell");
var TempDir = WshShell.ExpandEnvironmentStrings("%temp%") + "\\";
var mainFolder = profileRoot.getValue("Snegopat/MainFolder")

var СоответствиеФайловИСтатусов = [];

var мФормаНастройки=null

function macrosНастрокаFossil(){
    var pathToForm=SelfScript.fullPath.replace(/js$/, 'ssf')
    мФормаНастройки=loadScriptForm(pathToForm, SelfScript.self) // Обработку событий формы привяжем к самому скрипту
    мФормаНастройки.ОткрытьМодально()
}

function мЗаписатьНастройки()
{
    var FossilSnegopat = мФормаНастройки.cmdSnegopat;
    var FossilBase = мФормаНастройки.cmdBase;
    profileRoot.setValue(pfFossilPath, FossilSnegopat)
    profileRoot.setValue(pfFossilPathBase, FossilBase)
}

function НастройкиПриОткрытии()
{
    мФормаНастройки.cmdSnegopat=мPathToFossil
    мФормаНастройки.cmdBase=мPathToFossilBase
}

function cmdSnegopatНачалоВыбора(Элемент, СтандартнаяОбработка) {
    лФайл=мВыбратьФайл()
    if(лФайл=="") return
    Элемент.val.Значение=лФайл
}

function cmdBaseНачалоВыбора(Элемент, СтандартнаяОбработка) {
    лФайл=мВыбратьФайл()
    if(лФайл=="") return
    Элемент.val.Значение=лФайл
}

function КнопкаЗаписатьНажатие(Кнопка) {
    мЗаписатьНастройки();
    мФормаНастройки.Закрыть();
}

function мВыбратьФайл()
{
    ДиалогОткрытияФайла=v8New("ДиалогВыбораФайла", РежимДиалогаВыбораФайла.Открытие)
    //ДиалогОткрытияФайла.ПолноеИмяФайла = ""
    ДиалогОткрытияФайла.Заголовок = "Выберите файл с fossil "
    if(ДиалогОткрытияФайла.Выбрать()==false) return ""
    return ДиалогОткрытияФайла.ПолноеИмяФайла
}

function СоздатьРеопзиторийНажатие(Кнопка) { 
    ДиалогОткрытияФайла=v8New("ДиалогВыбораФайла", РежимДиалогаВыбораФайла.Save);
    ДиалогОткрытияФайла.Заголовок = "Выберите файл, для создания репозитария ";
    if(ДиалогОткрытияФайла.Выбрать()) {
        var PathToBatFossil = TempDir + "fossilTrue.bat"
        var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
        var TextDoc = v8New("TextDocument");
        TextDoc.Записать(PathToFossilOutput, "UTF-8");
        //var pathToCatalog = f.Path;
        //TextDoc.AddLine('cd /d"' +млКаталог +'"')
        TextDoc.AddLine(PathToFossil +' new "'+ДиалогОткрытияФайла.ПолноеИмяФайла+'"');
        TextDoc.AddLine("exit")
        TextDoc.Write(PathToBatFossil, 'cp866');
        ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 1, 1);
    }
}

function ОткрытьРепоНажатие(Кнопка) { 
    лФайл=мВыбратьФайл()
    if(лФайл=="") return
    ДиалогОткрытияФайла=v8New("ДиалогВыбораФайла", РежимДиалогаВыбораФайла.ВыборКаталога);
    ДиалогОткрытияФайла.Заголовок = "Выберите каталог, куда открыть репозиторий ";
    if(ДиалогОткрытияФайла.Выбрать()) {
        var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
        var PathToBatFossil = TempDir + "fossilTrue.bat"
        var TextDoc = v8New("TextDocument");
        TextDoc.Записать(PathToFossilOutput, "UTF-8");
        //var pathToCatalog = f.Path;
        TextDoc.AddLine('cd /d "' +ДиалогОткрытияФайла.Каталог +'"');
        TextDoc.AddLine(PathToFossil +' open "'+лФайл+'"');
        TextDoc.AddLine("exit")
        TextDoc.Write(PathToBatFossil, 'cp866');
        ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 1, 1);
    }
}

function ЗапуститьFossilНажатие(Кнопка) { 
    fossil_run(mainFolder);
}


function fossil_getRootCatalog(path){
    var result = "";
    for (var key in СоответствиеФайловИСтатусов){
        if (path.indexOf(key)!=-1) {
            result = key
            break
        }
    }
    if (result==undefined) {
        var f = v8New("File", path);
        if (f.Exist()) {
            var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
            var PathToBatFossil = TempDir + "fossilTrue.bat"
            var TextDoc = v8New("TextDocument");
            TextDoc.Записать(PathToFossilOutput, "UTF-8");
            //var pathToCatalog = f.Path;
            млКаталог = f.Path;
            TextDoc.AddLine('cd /d"' +млКаталог +'"')
            TextDoc.AddLine(PathToFossil +' status >> "'+PathToFossilOutput+'"');
            TextDoc.Write(PathToBatFossil, 'cp866');
            ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
            TextDoc.Read(PathToFossilOutput, "UTF-8");
            if (TextDoc.LineCount() == 0) {
                Message ("комманда отработала, но вывод не записался, надо отладить!")
                return "" //что то пошло не так. 
            }
        
            var i=0;
            for (var i=1; i<=TextDoc.LineCount(); i++)
            {
                var r = TextDoc.GetLine(i);
                if (r.indexOf("local-root:")!=-1){ // все нашли, теперь 
                    млКаталог  = r.split('   ')[1];
                    лКаталог = млКаталог.replace(/\//g, '\\');
                    лКаталог = лКаталог.substr(0, лКаталог.length-1);
                    break;
                }
            }
            TextDoc.Clear();
            TextDoc.Write(PathToFossilOutput, "UTF-8");
            result = лКаталог
        }
    }
    return result
} //fossil_getRootCatalog

function fossil_test(pathToCatalog) {
    
    var f = v8New("File", pathToCatalog);
    if (!f.Exist()) return false; 
    if (!f.IsDirectory()) return false;
    if (!FSO.FileExists(FSO.BuildPath(pathToCatalog, '_FOSSIL_'))) return false; //есть файл, тогда пробуем. Возможна проблема с тем, что _FOSSIL_ указывает у себя путь к несуществющему репозитарию.
    
    return true
} //fossil_test

function fossil_getStatusForCatalog(pathToCatalog, ValueTablesFiles) {

    СоответствиеФайловИСтатусов[pathToCatalog] = {};
    
    var СоответствиеСтатусов = СоответствиеФайловИСтатусов[pathToCatalog];
    var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
    var PathToBatFossil = TempDir + "fossilTrue.bat"
    var TextDoc = v8New("TextDocument");
    TextDoc.AddLine("Временный файл для fossil");
    TextDoc.Write(PathToFossilOutput, "UTF-8")
    var TextDoc = v8New("TextDocument");
    TextDoc.AddLine('cd /d "' +pathToCatalog+'"');
    TextDoc.AddLine(PathToFossil +' status >> "'+PathToFossilOutput+'"');
    TextDoc.AddLine('echo NOTVERSIONED>> "'+PathToFossilOutput+'"');
    TextDoc.AddLine(PathToFossil+' extras >> "'+PathToFossilOutput+'"');
    TextDoc.AddLine('echo ENDNOTVERSIONED>> "'+PathToFossilOutput+'"');
    TextDoc.Write(PathToBatFossil, "cp866");
    ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
    TextDoc.Read(PathToFossilOutput, "UTF-8");
    if (TextDoc.LineCount() == 0) {
        Message ("комманда отработала, но вывод не записался, надо отладить!")
        return false //что то пошло не так. 
    }
    var re = new RegExp(/(ADDED_BY_MERGE|UPDATED_BY_MERGE|DELETED_BY_MERGE|ADDED|EDITED|ADDED|MISSING|tags:)\s*(.*)\n/g)
    var r = TextDoc.ПолучитьТекст();
    var matches;
    var index=0;
    while ((matches = re.exec(r)) != null)
    {
        filename = matches[2];
        if (matches[1].indexOf('tags')!=-1){
            СоответствиеСтатусов[pathToCatalog]=filename;
            continue;
        }
        if (matches[1] == 'MISSING'){
            СоответствиеСтатусов[FSO.BuildPath(pathToCatalog, filename.replace(/\//g, '\\'))]= "DELETED"
            continue;
        }
        СоответствиеСтатусов[FSO.BuildPath(pathToCatalog, filename.replace(/\//g, '\\'))]= matches[1]
    }
    var re = new RegExp(/NOTVERSIONED\s*\n((.|\s)*?)ENDNOTVERSIONED/g);
    var re_notversionet = new RegExp(/^(.*)\n/gm);
    var matches;
    var index=0;
    while ((matches = re.exec(r)) != null)
    {
        var matches_comment;
        var text = matches[1]
        while ((matches_comment = re_notversionet.exec(text)) != null)
        {
            filename = matches_comment[1];
            СоответствиеСтатусов[FSO.BuildPath(pathToCatalog, filename.replace(/\//g, '\\'))]= "NOTVERSIONED"
        }
    }
    
    return true
} //fossil_getStatusForCatalog

function fossil_getFileAtRevision(pathToFile, rev){
    var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
    var PathToBatFossil = TempDir + "fossilTrue.bat"
    var TextDoc = v8New("TextDocument");
    TextDoc.Записать(PathToFossilOutput, "UTF-8");
    
    var f = v8New("File", pathToFile);
    if (!f.Exist()) 
    {
        Message(" файла физически не существует...") 
        return nell
    }
    var rootCatalog = fossil_getRootCatalog(pathToFile)
    if (rev.length !=0) {
    
        var filerev = FSO.BuildPath(TempDir, rev+f.Имя);
        
        
        TextDoc.AddLine('cd /d "' +rootCatalog +'"')
        TextDoc.AddLine(PathToFossil+' revert -r '+rev +' "'+pathToFile +'" ')
        TextDoc.AddLine('copy /Y "'+pathToFile +'" "'+filerev+'"')
        TextDoc.AddLine(PathToFossil+' undo "'+pathToFile +'" ')
        TextDoc.Write(PathToBatFossil, 'cp866');
        TextDoc.Clear();
        ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
        return filerev;
    }
    return null
} //fossil_getFileAtRevision

function fossil_getFilePathToDiff(param1, param2) { //текущая версия файла с предыдущей...
    
    var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
    var PathToBatFossil = TempDir + "fossilTrue.bat"
    var TextDoc = v8New("TextDocument");
    TextDoc.Записать(PathToFossilOutput, "UTF-8");
    // возвращать будем структру, path1 и path2 
    var pathToFile = param1;
    var rootCatalog = fossil_getRootCatalog(pathToFile);
    param2.insert("path1", pathToFile);
    param2.insert("path2", "");
    /* 

Пока не забыл, как нам вытащить вариант старый файла...
АлгоритмТаков:
fossil finfo -b test.txt
[CODE]
7704d33278 2012-02-07 Sosna 'blal'
[/CODE]
для первой ревизии 
fossil finfo -p -r 7704d33278 test.txt  > blabla.txt
для второй ревизи
fossil revert -r 7704d33278 test.txt
copy test.txt > blabla.txt
да, да все через одно место....
 */
    var r = ""; // Текущая строка прочитанная
    var ver1 = '' // Номер версии первого файла
    var ver1sha1 = '' //sha1 первого файла в базе fossil
    
    // Запусим shell и найдем версии файлов. 
    TextDoc.Clear();
    TextDoc.AddLine('cd /d "' +rootCatalog +'"')
    TextDoc.AddLine(PathToFossil+' finfo -b --limit 1 "'+pathToFile+'" > "' +PathToFossilOutput+'"');
    TextDoc.Write(PathToBatFossil, 'cp866');
    
    ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
    TextDoc.Read(PathToFossilOutput, "UTF-8");
    if (TextDoc.LineCount() == 0) {
        Message (" 1 комманда отработала, но вывод не записался, надо отладить!")
        return false //что то пошло не так. 
    }
    if (TextDoc.LineCount() > 0){
        var r = TextDoc.GetLine(1)
        if (r.indexOf("no history for file") >=0) {
            Message("Файл не находился под версионным контролем, не счем сравнивать.");
            param2.insert("path1", pathToFile);
            param2.insert("path2", "");
            return param2;
        }
        ver1 = r.split(' ')[0]
    }
    TextDoc.Clear();
    TextDoc.Write(PathToFossilOutput, "UTF-8");

    if (ver1 == null || ver1 == "") {Message("ver 1 не нашли ничего"); return ;}
    var file2ToDiff = fossil_getFileAtRevision(pathToFile, ver1);
    if (file2ToDiff == null || file2ToDiff == "") {Message("Не смогли получить файл нужной ревизии" + ver1); return ;}
    param2.insert("path2", file2ToDiff);
    
    return true
} //getFilePathToDiff

function fossil_add(pathToFile, param2) {
    var rootCatalog = fossil_getRootCatalog(pathToFile);
    var f = v8New("File", pathToFile);
    if (f.IsDirectory()) {
        pathToFile = '.'
    } else {
        pathToFile = '"'+pathToFile+'"'
    }
    var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
    var PathToBatFossil = TempDir + "fossilTrue.bat"
    var TextDoc = v8New("TextDocument");
    TextDoc.AddLine('cd /d "'+rootCatalog+'"')
    TextDoc.AddLine(PathToFossil +' add ' +pathToFile);
    TextDoc.Write(PathToBatFossil, 'cp866');
    
    TextDoc.Clear();
    ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
    return ErrCode
} //fossil_add

function fossil_run(pathToFile){
    var rootCatalog = fossil_getRootCatalog(pathToFile);
    var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
    var PathToBatFossil = TempDir + "fossilTrue.bat"
    var TextDoc = v8New("TextDocument");
    var abspath = FSO.GetAbsolutePathName(PathToFossil);
    var f = v8New('File', abspath);
    if (f.Exist()) {
        TextDoc.AddLine('PATH = %PATH%;'+f.Path+'"');
    }
    TextDoc.AddLine('cd /d "'+rootCatalog+'"')
    TextDoc.AddLine('start cmd.exe')
    TextDoc.Write(PathToBatFossil, 'cp866');
    ЗапуститьПриложение(PathToBatFossil, "", true);
    TextDoc = null;
} //fossil_run

function fossil_getFileStatus(pathToCatalog, pathToFile){
    var лКаталог = pathToCatalog
    if (СоответствиеФайловИСтатусов[лКаталог] == undefined) {
        Message("Get status for " + pathToCatalog + " "+СоответствиеФайловИСтатусов[лКаталог]);
        var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
        var PathToBatFossil = TempDir + "fossilTrue.bat"
        var TextDoc = v8New("TextDocument");
        TextDoc.Записать(PathToFossilOutput, "UTF-8");
        var лКаталог ="";
        //if (лКаталог == '') { //определим текущий ROOT каталог для fossil 
        var млКаталог = pathToCatalog;
        TextDoc.AddLine('cd /d"' +млКаталог +'"')
        TextDoc.AddLine(PathToFossil +' status > "'+PathToFossilOutput+'"');
        TextDoc.Write(PathToBatFossil, 'cp866');
        ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
        TextDoc.Read(PathToFossilOutput, "UTF-8");
        if (TextDoc.LineCount() == 0) {
            Message ("комманда отработала, но вывод не записался, надо отладить!")
            return null //что то пошло не так. 
        }
        var i=0;
        for (var i=1; i<=TextDoc.LineCount(); i++)
        {
            var r = TextDoc.GetLine(i);
            if (r.indexOf("local-root:")!=-1){ // все нашли, теперь 
                    млКаталог  = r.split('   ')[1];
                    лКаталог = млКаталог.replace(/\//g, '\\');
                    лКаталог = лКаталог.substr(0, лКаталог.length-1);
                    break;
            }
        }
        TextDoc.Clear();
        TextDoc.Write(PathToFossilOutput, "UTF-8");
    }
    
    СоответсвиеФайлов = СоответствиеФайловИСтатусов[лКаталог];
    if (СоответсвиеФайлов == undefined) return null 
    
    return (СоответсвиеФайлов[pathToFile] == undefined) ? null : СоответсвиеФайлов[pathToFile]
    
} //fossil_getFileStatus

function fossil_revert(pathToFile, ver) {

    var rootCatalog = fossil_getRootCatalog(pathToFile);
    var f = v8New("File", pathToFile);
    var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
    var PathToBatFossil = TempDir + "fossilTrue.bat"
    var TextDoc = v8New("TextDocument");
    TextDoc.Записать(PathToFossilOutput, "UTF-8");
    TextDoc.AddLine('cd /d"' +rootCatalog +'"')
    var cmd = (ver.length>0) ? ''+PathToFossil +' revert -r '+ver+' "' +pathToFile+'"' : ''+PathToFossil +' revert  "' +pathToFile+'"';
    TextDoc.AddLine(cmd);
    TextDoc.Write(PathToBatFossil, 'cp866');
    TextDoc.Clear();
    ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
    return ErrCode
} //fossil_revert

function fossil_delete(pathToFile) {
    var rootCatalog = fossil_getRootCatalog(pathToFile);
    //var f = v8New("File", pathToFile);
    var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
    var PathToBatFossil = TempDir + "fossilTrue.bat"
    var TextDoc = v8New("TextDocument");
    TextDoc.AddLine('cd /d "'+rootCatalog+'"')
    TextDoc.AddLine(PathToFossil +' dele "' +pathToFile+'"');
    TextDoc.Write(PathToBatFossil, 'cp866');
    TextDoc.Clear();
    ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
    return ErrCode
} //fossil_delete

function fossil_commit(pathToFile, message) {
    var rootCatalog = fossil_getRootCatalog(pathToFile);
    var tempfile = GetTempFileName("txt");
    var f = v8New("File", pathToFile);
    if (f.IsDirectory()) {
        pathToFile = ''
    } else {
        pathToFile = '"'+pathToFile+'"'
    }
    var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
    var PathToBatFossil = TempDir + "fossilTrue.bat"
    var TextDoc = v8New("TextDocument");
    TextDoc.AddLine('cd /d "'+rootCatalog+'"')
    TextDoc.AddLine(PathToFossil +' commit ' +pathToFile+' -M "'+tempfile+'"');
    TextDoc.AddLine('exit');
    TextDoc.Write(PathToBatFossil, 'cp866');
    
    TextDoc.Clear();
    TextDoc.SetText(message);
    TextDoc.Write(tempfile, 'utf-8');
    ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 1, 1);
    return ErrCode
} //fossil_commit

function fossil_showlog(pathToFile) { //временно, надо нарисовать красивю форму. 
    
    var rootCatalog = fossil_getRootCatalog(pathToFile);
    var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
    var PathToBatFossil = TempDir + "fossilTrue.bat"
    var TextDoc = v8New("TextDocument");
    TextDoc.AddLine('cd /d "'+rootCatalog+'"')
    TextDoc.AddLine(PathToFossil+' ui')
    TextDoc.Write(PathToBatFossil, 'cp866');
    ЗапуститьПриложение(PathToBatFossil);
} //fossil_showlog

function fossil_getLog(pathToFile, limit) { //если каталог, тогда информация для каталога, если файл, тогда лог для файла. 
    //Возвращаем массив со стурктурой:
    // arrary[0]['version':122333, 'comment':"Че то написали", 'author':"sosna", 'date':"2012-04-01"]
    var result = []
    //{ FIXME: умножаем кво коммитов на 2, считаем приблизительно 2 строки на 1 коммит.
    // в fossil есть ошибка вывода в коммандную строку кво. строк а не кво коммитов.
    // есть, даже патч
    // http://www.fossil-scm.org/index.html/info/3e58b8ceaf  
    //}
    limit = limit*2;
    
    f = v8New("File", pathToFile);
    if (!f.Exist()) return result
    //Проверим, есть ли он под версионным контролем у нас.
    var rootCatalog = fossil_getRootCatalog(pathToFile);
    var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
    var PathToBatFossil = TempDir + "fossilTrue.bat"
    var TextDoc = v8New("TextDocument");
    TextDoc.AddLine('cd /d "'+rootCatalog+'"')
    //пока будем нормально возвращать только для файла, надо спросить совета про парсинг общей истории...
    if (!f.IsDirectory()) { //Для файлов оставляем старый вариант, в timeline нет возможности отфильтровать сразу по файлам. 
        var ПутьОтносительноКорневогоКаталога = pathToFile.replace(rootCatalog+'\\', '');
        TextDoc.AddLine(PathToFossil+' finfo -l --limit '+limit+' "'+ПутьОтносительноКорневогоКаталога +'" > "'+PathToFossilOutput+'"')
        TextDoc.Write(PathToBatFossil, 'cp866');
        
        ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
        TextDoc.Clear();

        TextDoc.Read(PathToFossilOutput, "UTF-8");
        if (TextDoc.LineCount() == 0) {
            return result 
        }
        
        re = new RegExp(/(\d{4}-\d{2}-\d{2})\s\[([0-9a-f]{10})\]\s((.|\s)*?)\(user:\s+(.+),\s+artifact/g);
        
        var r = TextDoc.ПолучитьТекст();
        var matches;
        var index=0;
        while ((matches = re.exec(r)) != null)
        {
            result[index] = {"version":matches[2], "comment":''+matches[3].replace(/\s{2,}|\n/g, " "), "date":matches[1], "author":matches[5]}
            index++;
        }
    } else { 
        TextDoc.AddLine(PathToFossil+' json timeline checkin --limit '+limit+'  >'+' "'+PathToFossilOutput+'"')
        TextDoc.Write(PathToBatFossil, 'cp866');
        
        ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
        TextDoc.Clear();
        TextDoc.Read(PathToFossilOutput, "UTF-8");
        debugger
        jsonoutput = TextDoc.GetText();
        jsonobject = eval('('+jsonoutput+')');
        if (!(jsonobject.resultCode == undefined))
            return result
        if (TextDoc.LineCount() == 0) {
            return result 
        }
        if (jsonobject.payload.timeline == undefined) 
            return result
        
        for (var i=0; i<jsonobject.payload.timeline.length; i++) {
            var timeline = jsonobject.payload.timeline[i];
            var date = new Date(timeline.timestamp);
            result[i] = {"version":timeline.uuid, "comment":'('+timeline.tags[0]+')'+' '+timeline.comment, "date":'' +date.toString(), "author":timeline.user}
        }
        
        // var re = new RegExp(/===\s((20\d\d)-(0[1-9]|2[012])-(0[1-9]|[12][0-9]|3[01]))\s===((\n(([01][0-9]|2[0-3]):([0-5][0-9]):([0-5][0-9]))\s\[([0-9a-f]{10})\]\s((.|\s)*?)\(user:\s+(.+)\s+tags:\s+(\w+)\))+)/g)
        // var re_comment = new RegExp(/(([01][0-9]|2[0-3]):([0-5][0-9]):([0-5][0-9]))\s\[([0-9a-f]{10})\]\s((.|\s)*?)\(user:\s+(.+)\s+tags:\s+(\w+)\)/g);
        // var r = TextDoc.ПолучитьТекст();
        // var matches;
        // var index=0;
        //debugger;
        // while ((matches = re.exec(r)) != null)
        // {
            // var matches_comment;
            // var text = matches[5]
            // while ((matches_comment = re_comment.exec(text)) != null)
            // {
                // /* var cmd = "";
                // for (var i=1; i < matches_comment.length; i++)
                    // var cmd = cmd + " "+i+" - "+matches_comment[i] */
                
                // result[index] = {"version":matches_comment[5], "comment":'('+matches_comment[9]+')'+' '+matches_comment[6], "date":'' +matches[1]+' '+matches_comment[1], "author":matches_comment[8]}
                // index++;
            // }
        // }
    }
    
return result;    
} // fossil_getLog

function fossil_getInfo(pathToFile, ver) {
    var result = {"comment":"", "files":[]}
    var rootCatalog = fossil_getRootCatalog(pathToFile);
    var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
    var PathToBatFossil = TempDir + "fossilTrue.bat"
    var TextDoc = v8New("TextDocument");
    TextDoc.AddLine('cd /d "'+rootCatalog+'"')
    //var ПутьОтносительноКорневогоКаталога = pathToFile.replace(rootCatalog+'\\', '');
    TextDoc.AddLine(PathToFossil+' info  '+ver +' > "'+PathToFossilOutput+'"')
    TextDoc.Write(PathToBatFossil, 'cp866');
    ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
    TextDoc.Clear();
    TextDoc.Read(PathToFossilOutput, "UTF-8");
    result["comment"] = TextDoc.GetText();
    TextDoc.Clear();
    TextDoc.AddLine('cd /d "'+rootCatalog+'"')
    TextDoc.AddLine(PathToFossil+' timeline '+ver +' -n 1 -showfiles -t ci > "'+PathToFossilOutput+'"')
    TextDoc.Write(PathToBatFossil, 'cp866');
    ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
    TextDoc.Clear();
    TextDoc.Read(PathToFossilOutput, "UTF-8");
    if (TextDoc.LineCount() == 0) {
        //Message(" 0");
        return result 
    }
    var index=0;
    for (var i=1; i<=TextDoc.LineCount(); i++)
    {
        var r = TextDoc.GetLine(i);
        /* if (r.length > 0) && (r[0]=='F') { //файл
            var ar = r.split(' ');
        } */
        //if (r.indexOf("file outside of")!=-1) return result
        //re = new RegExp(/(F)\s(\S*)\s(\S*)/);
        re = new RegExp(/\s*(EDITED|ADDED|DELETED)\s(.*)/);
        var mathes = r.match(re);
        //Message(" 1"); 
        if (mathes && mathes.length) {
            
            //Message(" 2" + mathes[1]);
             //млКаталог.replace(/\//g, '\\')
            fullpathfile = mathes[2].replace(/\\s/g, ' ') //пробел так fossil отображает.
            //fullpathfile = fullpathfile.replace(/\//g, '\\');
            result['files'][index] = {"version":ver, "file":''+fullpathfile, "status":mathes[1], "fullpath":FSO.BuildPath(rootCatalog, fullpathfile.replace(/\//g, '\\'))}
            index++;
        }
    }
    return result
}

function fossil_getListBranch(pathToFile, index) {
    
    result = {"valuelist":v8New("ValueList"), "index":-1}
    //result = v8New("ValueList");
    var rootCatalog = fossil_getRootCatalog(pathToFile);
    var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
    var PathToBatFossil = TempDir + "fossilTrue.bat"
    var TextDoc = v8New("TextDocument");
    TextDoc.AddLine('cd /d "'+rootCatalog+'"')
    //var ПутьОтносительноКорневогоКаталога = pathToFile.replace(rootCatalog+'\\', '');
    TextDoc.AddLine(PathToFossil+' branch  > "'+PathToFossilOutput+'"')
    TextDoc.Write(PathToBatFossil, 'cp866');
    ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
    TextDoc.Clear();
    TextDoc.Read(PathToFossilOutput, "UTF-8");
    var re = new RegExp(/(\s*|\*\s)(\S*)\n/g);
    var r = TextDoc.ПолучитьТекст();
    var matches;
    var index=0;
    //debugger;
    while ((matches = re.exec(r)) != null)
    {
        result['valuelist'].add(matches[2], matches[2])
        if (matches[1].indexOf("\*")!=-1) result["index"]=index;
        index++;
    }
    //Добавим сюда комманды создания новой ветки...
    result['valuelist'].add("NEWBRANCH", "Новая ветка");
    result['valuelist'].add("NEWBRANCHPRIVATE", "Новая приватная ветка");
    
    return result;
    
}

function fossil_swithBranch (pathToFile, branch) {
    
    if ((branch == "NEWBRANCH") || (branch == "NEWBRANCHPRIVATE")) return fossil_createBranch(pathToFile, branch);
    var rootCatalog = fossil_getRootCatalog(pathToFile);
    var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
    var PathToBatFossil = TempDir + "fossilTrue.bat"
    var TextDoc = v8New("TextDocument");
    TextDoc.AddLine('cd /d "'+rootCatalog+'"')
    //var ПутьОтносительноКорневогоКаталога = pathToFile.replace(rootCatalog+'\\', '');
    TextDoc.AddLine(PathToFossil+' update '+ branch +' > "'+PathToFossilOutput+'"');
    TextDoc.Write(PathToBatFossil, 'cp866');
    ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
    TextDoc.Clear();
    return true;
}

function fossil_createBranch(pathToFile, branch, type) {
    
    var name = "";
    var checkin = "";
    if (type == undefined) type = '';
    if (branch == "NEWBRANCHPRIVATE") {
        
        type = "--private";
    }
    
    if ((branch == "NEWBRANCH") || (branch == "NEWBRANCHPRIVATE")){
        var vbs = addins.byUniqueName("vbs").object
        vbs.var0 = ""; vbs.var1 = "Введите имя ветки"; vbs.var2 = 0, vbs.var3 = false;
        if (vbs.DoEval("InputString(var0, var1, var2, var3)")) {
            var message  = vbs.var0;
            name = message;
        }
    } else {
        name = branch;
    }
    
    if (name.length==0) {
        Message('Имя ветки для создания пустое.');
        return false;
    }
    
    var rootCatalog = fossil_getRootCatalog(pathToFile);
    var PathToFossilOutput = TempDir + "fossilstatus.txt" // Пишем 1С файл в utf-8, выводим туда статус fossil после этого читаем его. 
    var PathToBatFossil = TempDir + "fossilTrue.bat"
    var TextDoc = v8New("TextDocument");
    TextDoc.AddLine('cd /d "'+rootCatalog+'"');
    TextDoc.AddLine(PathToFossil +' status > "'+PathToFossilOutput+'"');
    TextDoc.Write(PathToBatFossil, 'cp866');
    ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 0, 1)
    TextDoc.Clear();
    TextDoc.Read(PathToFossilOutput, "UTF-8");
    var re = new RegExp(/^checkout:\s*(\w*)\s*/m);
    var r = TextDoc.ПолучитьТекст();
    var matches = r.match(re);
    
    if (matches && matches.length) {
        checkin = matches[1];
    }
    
    TextDoc.AddLine('cd /d "'+rootCatalog+'"');
    //TextDoc.AddLine(PathToFossil +' branch new '+name+' '+checkin+' '+type+' > "'+PathToFossilOutput+'"');
    TextDoc.AddLine(PathToFossil +' branch new '+name+' '+checkin+' '+type);
    TextDoc.AddLine('exit');
    TextDoc.Write(PathToBatFossil, 'cp866');
    ErrCode = WshShell.Run('"'+PathToBatFossil+'"', 1, 1);
    TextDoc.Clear();
    fossil_swithBranch(pathToFile, name);
    
    return true;
}

function fossil_getMissingFiles(pathToFile) {
    var лКаталог = pathToFile
    var result = []
    if (СоответствиеФайловИСтатусов[лКаталог] == undefined) return result
    СоответсвиеФайлов = СоответствиеФайловИСтатусов[лКаталог];
    if (СоответсвиеФайлов == undefined) return result
    var index = 0;
    for (var key in СоответсвиеФайлов){
        if (СоответсвиеФайлов[key]=="DELETED") {
            result[index] = key.replace(лКаталог+"\\", '');
            index++;
        }
    }
    return result
}


function Backend_fossil(command, param1, param2) {
    var result = false;
    switch (command) 
    {
    case "CATALOGSTATUS":
        // Добавляем в хвост подпись.
        result = fossil_getStatusForCatalog(param1, "");
        break;
    case "FILESTATUS":
        result = fossil_getFileStatus(param1, param2)
        break;
    case "GETFILESDELETED":
        result = fossil_getMissingFiles(param1);
        break;
    case "DIFF":
        result = fossil_getFilePathToDiff(param1, param2)
        break;
    case "ADD":
        result = fossil_add(param1, param2)
        break;
    case "TEST":
        result = fossil_test(param1)
        break;
    case "RUN":
        result = fossil_run(param1, param2)
        break;
    case "SHOWLOG":
        result = fossil_showlog(param1);
        break
    case "SHOWDIFF":
        result = fossil_getFilePathToDiff(param1, param2);
        break
    case "DELETE":
        result = fossil_delete(param1)
        break
    case "REVERT":
        result = fossil_revert(param1, param2);
        break
    case "COMMIT":
        result = fossil_commit(param1, param2);
        break
    case "GETFILEATREVISION":
        result = fossil_getFileAtRevision(param1, param2)
        break
    case "GETLOG":
        result = fossil_getLog(param1, param2);
        break
    case "GETINFO":
        result = fossil_getInfo(param1, param2);
        break
    case "GETLISTBRANCH":
        result = fossil_getListBranch(param1); //возвращаем result {"valuelist":v8New("ValueList"), "index": индекс ветки текущей}
        break
    case "SWITHBRANCH":
        result = fossil_swithBranch(param1, param2);
        break;
    }
    return result
} //Backend_fossil

function GetBackend() {
    return Backend_fossil
} //GetBackend

function getDefaultMacros() {
    return 'НастрокаFossil'
} //getDefaultMacros