2017-09-20 11:30:07 +02:00
|
|
|
/**
|
|
|
|
* Copyright JS Foundation and other contributors, http://js.foundation
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
**/
|
|
|
|
|
|
|
|
var fs = require('fs-extra');
|
|
|
|
var when = require('when');
|
|
|
|
var fspath = require("path");
|
|
|
|
var nodeFn = require('when/node/function');
|
|
|
|
var crypto = require('crypto');
|
|
|
|
|
|
|
|
var storageSettings = require("../settings");
|
|
|
|
var util = require("../util");
|
|
|
|
var gitTools = require("./git");
|
|
|
|
|
2017-10-17 00:23:50 +02:00
|
|
|
var Projects = require("./Project");
|
|
|
|
|
2017-09-20 11:30:07 +02:00
|
|
|
var settings;
|
|
|
|
var runtime;
|
|
|
|
|
|
|
|
var projectsDir;
|
2017-10-17 00:23:50 +02:00
|
|
|
var activeProject
|
2017-09-20 11:30:07 +02:00
|
|
|
|
|
|
|
function init(_settings, _runtime) {
|
|
|
|
settings = _settings;
|
|
|
|
runtime = _runtime;
|
|
|
|
log = runtime.log;
|
2017-11-22 00:31:41 +01:00
|
|
|
gitTools.init(_settings, _runtime);
|
2017-09-20 11:30:07 +02:00
|
|
|
|
2017-10-17 00:23:50 +02:00
|
|
|
Projects.init(settings,runtime);
|
|
|
|
|
2017-09-20 11:30:07 +02:00
|
|
|
projectsDir = fspath.join(settings.userDir,"projects");
|
|
|
|
|
|
|
|
if (settings.flowFile) {
|
|
|
|
flowsFile = settings.flowFile;
|
|
|
|
// handle Unix and Windows "C:\"
|
|
|
|
if ((flowsFile[0] == "/") || (flowsFile[1] == ":")) {
|
|
|
|
// Absolute path
|
|
|
|
flowsFullPath = flowsFile;
|
|
|
|
} else if (flowsFile.substring(0,2) === "./") {
|
|
|
|
// Relative to cwd
|
|
|
|
flowsFullPath = fspath.join(process.cwd(),flowsFile);
|
|
|
|
} else {
|
|
|
|
try {
|
|
|
|
fs.statSync(fspath.join(process.cwd(),flowsFile));
|
|
|
|
// Found in cwd
|
|
|
|
flowsFullPath = fspath.join(process.cwd(),flowsFile);
|
|
|
|
} catch(err) {
|
|
|
|
// Use userDir
|
|
|
|
flowsFullPath = fspath.join(settings.userDir,flowsFile);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
flowsFile = 'flows_'+require('os').hostname()+'.json';
|
|
|
|
flowsFullPath = fspath.join(settings.userDir,flowsFile);
|
|
|
|
}
|
|
|
|
var ffExt = fspath.extname(flowsFullPath);
|
|
|
|
var ffBase = fspath.basename(flowsFullPath,ffExt);
|
|
|
|
|
|
|
|
flowsFileBackup = getBackupFilename(flowsFullPath);
|
|
|
|
credentialsFile = fspath.join(settings.userDir,ffBase+"_cred"+ffExt);
|
|
|
|
credentialsFileBackup = getBackupFilename(credentialsFile)
|
|
|
|
|
|
|
|
if (!settings.readOnly) {
|
2017-10-07 01:18:20 +02:00
|
|
|
return fs.ensureDir(projectsDir)
|
2017-09-20 11:30:07 +02:00
|
|
|
//TODO: this is accessing settings from storage directly as settings
|
|
|
|
// has not yet been initialised. That isn't ideal - can this be deferred?
|
|
|
|
.then(storageSettings.getSettings)
|
|
|
|
.then(function(globalSettings) {
|
|
|
|
if (!globalSettings.projects) {
|
|
|
|
// TODO: Migration Case
|
|
|
|
console.log("TODO: Migration from single file to project");
|
|
|
|
globalSettings.projects = {
|
|
|
|
activeProject: "",
|
|
|
|
projects: {}
|
|
|
|
}
|
|
|
|
return storageSettings.saveSettings(globalSettings);
|
|
|
|
} else {
|
|
|
|
activeProject = globalSettings.projects.activeProject;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
return when.resolve();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function getBackupFilename(filename) {
|
|
|
|
var ffName = fspath.basename(filename);
|
|
|
|
var ffDir = fspath.dirname(filename);
|
|
|
|
return fspath.join(ffDir,"."+ffName+".backup");
|
|
|
|
}
|
|
|
|
|
2017-10-17 00:23:50 +02:00
|
|
|
function loadProject(name) {
|
|
|
|
return Projects.get(name).then(function(project) {
|
|
|
|
activeProject = project;
|
|
|
|
flowsFullPath = project.getFlowFile();
|
|
|
|
flowsFileBackup = project.getFlowFileBackup();
|
|
|
|
credentialsFile = project.getCredentialsFile();
|
|
|
|
credentialsFileBackup = project.getCredentialsFileBackup();
|
|
|
|
return project;
|
2017-09-20 11:30:07 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-10-17 00:23:50 +02:00
|
|
|
function getProject(name) {
|
2017-10-17 11:14:50 +02:00
|
|
|
checkActiveProject(name);
|
2017-10-17 00:23:50 +02:00
|
|
|
//return when.resolve(activeProject.info);
|
2017-10-17 11:14:50 +02:00
|
|
|
return Projects.get(name);
|
2017-09-20 11:30:07 +02:00
|
|
|
}
|
|
|
|
|
2017-10-17 11:14:50 +02:00
|
|
|
function checkActiveProject(project) {
|
|
|
|
if (!activeProject || activeProject.name !== project) {
|
|
|
|
//TODO: throw better err
|
2017-11-22 00:31:41 +01:00
|
|
|
throw new Error("Cannot operate on inactive project wanted:"+project+" current:"+(activeProject&&activeProject.name));
|
2017-10-17 11:14:50 +02:00
|
|
|
}
|
|
|
|
}
|
2017-09-20 11:30:07 +02:00
|
|
|
function getFiles(project) {
|
2017-10-17 11:14:50 +02:00
|
|
|
checkActiveProject(project);
|
2017-10-17 00:23:50 +02:00
|
|
|
return activeProject.getFiles();
|
2017-09-20 11:30:07 +02:00
|
|
|
}
|
2017-10-07 01:18:20 +02:00
|
|
|
function stageFile(project,file) {
|
2017-10-17 11:14:50 +02:00
|
|
|
checkActiveProject(project);
|
2017-10-17 00:23:50 +02:00
|
|
|
return activeProject.stageFile(file);
|
2017-10-07 01:18:20 +02:00
|
|
|
}
|
|
|
|
function unstageFile(project,file) {
|
2017-10-17 11:14:50 +02:00
|
|
|
checkActiveProject(project);
|
2017-10-17 00:23:50 +02:00
|
|
|
return activeProject.unstageFile(file);
|
2017-10-07 01:18:20 +02:00
|
|
|
}
|
|
|
|
function commit(project,options) {
|
2017-10-17 11:14:50 +02:00
|
|
|
checkActiveProject(project);
|
2017-10-17 00:23:50 +02:00
|
|
|
return activeProject.commit(options);
|
2017-09-20 11:30:07 +02:00
|
|
|
}
|
2017-10-09 01:11:07 +02:00
|
|
|
function getFileDiff(project,file,type) {
|
2017-10-17 11:14:50 +02:00
|
|
|
checkActiveProject(project);
|
2017-10-17 00:23:50 +02:00
|
|
|
return activeProject.getFileDiff(file,type);
|
2017-10-09 01:11:07 +02:00
|
|
|
}
|
2017-10-10 00:37:19 +02:00
|
|
|
function getCommits(project,options) {
|
2017-10-17 11:14:50 +02:00
|
|
|
checkActiveProject(project);
|
2017-10-17 00:23:50 +02:00
|
|
|
return activeProject.getCommits(options);
|
2017-10-10 00:37:19 +02:00
|
|
|
}
|
|
|
|
function getCommit(project,sha) {
|
2017-10-17 11:14:50 +02:00
|
|
|
checkActiveProject(project);
|
2017-10-17 00:23:50 +02:00
|
|
|
return activeProject.getCommit(sha);
|
2017-10-10 00:37:19 +02:00
|
|
|
}
|
|
|
|
|
2017-10-25 16:26:24 +02:00
|
|
|
function getFile(project,filePath,sha) {
|
|
|
|
checkActiveProject(project);
|
|
|
|
return activeProject.getFile(filePath,sha);
|
2017-09-20 11:30:07 +02:00
|
|
|
}
|
2017-11-22 00:31:41 +01:00
|
|
|
function push(project,remoteBranchName,setRemote) {
|
|
|
|
checkActiveProject(project);
|
|
|
|
return activeProject.push(remoteBranchName,setRemote);
|
|
|
|
}
|
|
|
|
function pull(project,remoteBranchName,setRemote) {
|
|
|
|
checkActiveProject(project);
|
|
|
|
return activeProject.pull(remoteBranchName,setRemote).then(reloadActiveProject);
|
|
|
|
}
|
|
|
|
function getStatus(project) {
|
|
|
|
checkActiveProject(project);
|
|
|
|
return activeProject.status();
|
|
|
|
}
|
|
|
|
function resolveMerge(project,file,resolution) {
|
|
|
|
checkActiveProject(project);
|
|
|
|
return activeProject.resolveMerge(file,resolution);
|
|
|
|
}
|
|
|
|
function abortMerge(project) {
|
|
|
|
checkActiveProject(project);
|
|
|
|
return activeProject.abortMerge().then(reloadActiveProject);
|
|
|
|
}
|
|
|
|
function getBranches(project,remote) {
|
|
|
|
checkActiveProject(project);
|
|
|
|
return activeProject.getBranches(remote);
|
|
|
|
}
|
|
|
|
function setBranch(project,branchName,isCreate) {
|
|
|
|
checkActiveProject(project);
|
|
|
|
return activeProject.setBranch(branchName,isCreate).then(reloadActiveProject);
|
|
|
|
}
|
|
|
|
function getBranchStatus(project,branchName) {
|
|
|
|
checkActiveProject(project);
|
|
|
|
return activeProject.getBranchStatus(branchName);
|
|
|
|
}
|
2017-09-20 11:30:07 +02:00
|
|
|
function getActiveProject() {
|
|
|
|
return activeProject;
|
|
|
|
}
|
|
|
|
|
2017-10-17 00:23:50 +02:00
|
|
|
function reloadActiveProject() {
|
2017-09-20 11:30:07 +02:00
|
|
|
return runtime.nodes.stopFlows().then(function() {
|
|
|
|
return runtime.nodes.loadFlows(true).then(function() {
|
2017-10-17 00:23:50 +02:00
|
|
|
runtime.events.emit("runtime-event",{id:"project-change",payload:{ project: activeProject.name}});
|
2017-10-07 01:18:20 +02:00
|
|
|
}).catch(function(err) {
|
2017-09-20 11:30:07 +02:00
|
|
|
// We're committed to the project change now, so notify editors
|
|
|
|
// that it has changed.
|
2017-10-17 00:23:50 +02:00
|
|
|
runtime.events.emit("runtime-event",{id:"project-change",payload:{ project: activeProject.name}});
|
2017-09-20 11:30:07 +02:00
|
|
|
throw err;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
2017-10-17 00:23:50 +02:00
|
|
|
function createProject(metadata) {
|
|
|
|
return Projects.create(metadata).then(function(p) {
|
2017-10-17 11:14:50 +02:00
|
|
|
return setActiveProject(p.name);
|
|
|
|
}).then(function() {
|
|
|
|
return getProject(metadata.name);
|
2017-10-17 00:23:50 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
function setActiveProject(projectName) {
|
|
|
|
return loadProject(projectName).then(function(project) {
|
2017-09-20 11:30:07 +02:00
|
|
|
var globalProjectSettings = settings.get("projects");
|
2017-10-17 00:23:50 +02:00
|
|
|
globalProjectSettings.activeProject = project.name;
|
2017-09-20 11:30:07 +02:00
|
|
|
return settings.set("projects",globalProjectSettings).then(function() {
|
|
|
|
log.info(log._("storage.localfilesystem.changing-project",{project:activeProject||"none"}));
|
|
|
|
log.info(log._("storage.localfilesystem.flows-file",{path:flowsFullPath}));
|
|
|
|
|
2017-10-17 00:23:50 +02:00
|
|
|
// console.log("Updated file targets to");
|
|
|
|
// console.log(flowsFullPath)
|
|
|
|
// console.log(credentialsFile)
|
|
|
|
return reloadActiveProject();
|
2017-09-20 11:30:07 +02:00
|
|
|
})
|
|
|
|
});
|
|
|
|
}
|
|
|
|
function updateProject(project,data) {
|
2017-10-17 00:23:50 +02:00
|
|
|
if (!activeProject || activeProject.name !== project) {
|
|
|
|
// TODO standardise
|
|
|
|
throw new Error("Cannot update inactive project");
|
|
|
|
}
|
2017-10-19 22:38:53 +02:00
|
|
|
// In case this triggers a credential secret change
|
|
|
|
var isReset = data.resetCredentialSecret;
|
|
|
|
var wasInvalid = activeProject.credentialSecretInvalid;
|
|
|
|
|
|
|
|
return activeProject.update(data).then(function(result) {
|
|
|
|
|
|
|
|
if (result.flowFilesChanged) {
|
|
|
|
flowsFullPath = activeProject.getFlowFile();
|
|
|
|
flowsFileBackup = activeProject.getFlowFileBackup();
|
|
|
|
credentialsFile = activeProject.getCredentialsFile();
|
|
|
|
credentialsFileBackup = activeProject.getCredentialsFileBackup();
|
|
|
|
return reloadActiveProject();
|
|
|
|
} else if (result.credentialSecretChanged) {
|
|
|
|
if (isReset || !wasInvalid) {
|
|
|
|
if (isReset) {
|
|
|
|
runtime.nodes.clearCredentials();
|
|
|
|
}
|
|
|
|
runtime.nodes.setCredentialSecret(activeProject.credentialSecret);
|
|
|
|
return runtime.nodes.exportCredentials()
|
|
|
|
.then(runtime.storage.saveCredentials)
|
|
|
|
.then(function() {
|
|
|
|
if (wasInvalid) {
|
|
|
|
return reloadActiveProject();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else if (wasInvalid) {
|
|
|
|
return reloadActiveProject();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2017-10-17 00:23:50 +02:00
|
|
|
}
|
|
|
|
function setCredentialSecret(data) { //existingSecret,secret) {
|
|
|
|
var isReset = data.resetCredentialSecret;
|
2017-10-17 11:14:50 +02:00
|
|
|
var wasInvalid = activeProject.credentialSecretInvalid;
|
2017-10-17 00:23:50 +02:00
|
|
|
return activeProject.update(data).then(function() {
|
|
|
|
if (isReset || !wasInvalid) {
|
|
|
|
if (isReset) {
|
|
|
|
runtime.nodes.clearCredentials();
|
|
|
|
}
|
|
|
|
runtime.nodes.setCredentialSecret(activeProject.credentialSecret);
|
|
|
|
return runtime.nodes.exportCredentials()
|
|
|
|
.then(runtime.storage.saveCredentials)
|
|
|
|
.then(function() {
|
|
|
|
if (wasInvalid) {
|
|
|
|
return reloadActiveProject();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else if (wasInvalid) {
|
|
|
|
return reloadActiveProject();
|
2017-09-20 11:30:07 +02:00
|
|
|
}
|
2017-10-17 00:23:50 +02:00
|
|
|
})
|
2017-09-20 11:30:07 +02:00
|
|
|
}
|
|
|
|
|
2017-10-17 00:23:50 +02:00
|
|
|
|
2017-09-20 11:30:07 +02:00
|
|
|
var initialFlowLoadComplete = false;
|
|
|
|
|
|
|
|
var flowsFile;
|
|
|
|
var flowsFullPath;
|
|
|
|
var flowsFileBackup;
|
|
|
|
var credentialsFile;
|
|
|
|
var credentialsFileBackup;
|
|
|
|
|
|
|
|
function getFlows() {
|
|
|
|
if (!initialFlowLoadComplete) {
|
|
|
|
initialFlowLoadComplete = true;
|
|
|
|
log.info(log._("storage.localfilesystem.user-dir",{path:settings.userDir}));
|
2017-10-07 01:18:20 +02:00
|
|
|
if (activeProject) {
|
2017-10-17 00:23:50 +02:00
|
|
|
return loadProject(activeProject).then(function() {
|
|
|
|
log.info(log._("storage.localfilesystem.active-project",{project:activeProject.name||"none"}));
|
|
|
|
log.info(log._("storage.localfilesystem.flows-file",{path:flowsFullPath}));
|
|
|
|
return getFlows();
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
log.info(log._("storage.localfilesystem.flows-file",{path:flowsFullPath}));
|
2017-10-07 01:18:20 +02:00
|
|
|
}
|
2017-09-20 11:30:07 +02:00
|
|
|
}
|
2017-10-19 22:38:53 +02:00
|
|
|
if (activeProject) {
|
|
|
|
if (!activeProject.getFlowFile()) {
|
|
|
|
log.warn("NLS: project has no flow file");
|
|
|
|
var error = new Error("NLS: project has no flow file");
|
|
|
|
error.code = "missing_flow_file";
|
|
|
|
return when.reject(error);
|
|
|
|
}
|
|
|
|
}
|
2017-09-20 11:30:07 +02:00
|
|
|
return util.readFile(flowsFullPath,flowsFileBackup,[],'flow');
|
|
|
|
}
|
|
|
|
|
|
|
|
function saveFlows(flows) {
|
|
|
|
if (settings.readOnly) {
|
|
|
|
return when.resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
fs.renameSync(flowsFullPath,flowsFileBackup);
|
|
|
|
} catch(err) {
|
|
|
|
}
|
|
|
|
|
|
|
|
var flowData;
|
|
|
|
|
|
|
|
if (settings.flowFilePretty) {
|
|
|
|
flowData = JSON.stringify(flows,null,4);
|
|
|
|
} else {
|
|
|
|
flowData = JSON.stringify(flows);
|
|
|
|
}
|
|
|
|
return util.writeFile(flowsFullPath, flowData);
|
|
|
|
}
|
|
|
|
|
|
|
|
function getCredentials() {
|
|
|
|
return util.readFile(credentialsFile,credentialsFileBackup,{},'credentials');
|
|
|
|
}
|
|
|
|
|
|
|
|
function saveCredentials(credentials) {
|
|
|
|
if (settings.readOnly) {
|
|
|
|
return when.resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
fs.renameSync(credentialsFile,credentialsFileBackup);
|
|
|
|
} catch(err) {
|
|
|
|
}
|
|
|
|
var credentialData;
|
|
|
|
if (settings.flowFilePretty) {
|
|
|
|
credentialData = JSON.stringify(credentials,null,4);
|
|
|
|
} else {
|
|
|
|
credentialData = JSON.stringify(credentials);
|
|
|
|
}
|
|
|
|
return util.writeFile(credentialsFile, credentialData);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
init: init,
|
2017-10-17 00:23:50 +02:00
|
|
|
listProjects: Projects.list,
|
2017-09-20 11:30:07 +02:00
|
|
|
getActiveProject: getActiveProject,
|
|
|
|
setActiveProject: setActiveProject,
|
|
|
|
getProject: getProject,
|
|
|
|
createProject: createProject,
|
|
|
|
updateProject: updateProject,
|
|
|
|
getFiles: getFiles,
|
2017-10-25 16:26:24 +02:00
|
|
|
getFile: getFile,
|
2017-10-07 01:18:20 +02:00
|
|
|
stageFile: stageFile,
|
|
|
|
unstageFile: unstageFile,
|
|
|
|
commit: commit,
|
2017-10-09 01:11:07 +02:00
|
|
|
getFileDiff: getFileDiff,
|
2017-10-10 00:37:19 +02:00
|
|
|
getCommits: getCommits,
|
|
|
|
getCommit: getCommit,
|
2017-11-22 00:31:41 +01:00
|
|
|
push: push,
|
|
|
|
pull: pull,
|
|
|
|
getStatus:getStatus,
|
|
|
|
resolveMerge: resolveMerge,
|
|
|
|
abortMerge: abortMerge,
|
|
|
|
getBranches: getBranches,
|
|
|
|
setBranch: setBranch,
|
|
|
|
getBranchStatus:getBranchStatus,
|
2017-09-20 11:30:07 +02:00
|
|
|
|
|
|
|
getFlows: getFlows,
|
|
|
|
saveFlows: saveFlows,
|
|
|
|
getCredentials: getCredentials,
|
|
|
|
saveCredentials: saveCredentials
|
|
|
|
|
|
|
|
};
|