2023-04-19 08:46:10 -07:00
|
|
|
import { Service } from 'typedi';
|
2023-05-31 06:01:57 -07:00
|
|
|
import path from 'path';
|
2023-04-19 08:46:10 -07:00
|
|
|
import * as Db from '@/Db';
|
2023-06-20 10:13:18 -07:00
|
|
|
import { sourceControlFoldersExistCheck } from './sourceControlHelper.ee';
|
|
|
|
import type { SourceControlPreferences } from './types/sourceControlPreferences';
|
2023-05-31 06:01:57 -07:00
|
|
|
import {
|
2023-06-20 10:13:18 -07:00
|
|
|
SOURCE_CONTROL_CREDENTIAL_EXPORT_FOLDER,
|
|
|
|
SOURCE_CONTROL_GIT_FOLDER,
|
|
|
|
SOURCE_CONTROL_README,
|
|
|
|
SOURCE_CONTROL_SSH_FOLDER,
|
|
|
|
SOURCE_CONTROL_SSH_KEY_NAME,
|
|
|
|
SOURCE_CONTROL_TAGS_EXPORT_FILE,
|
|
|
|
SOURCE_CONTROL_VARIABLES_EXPORT_FILE,
|
|
|
|
SOURCE_CONTROL_WORKFLOW_EXPORT_FOLDER,
|
2023-05-31 06:01:57 -07:00
|
|
|
} from './constants';
|
|
|
|
import { LoggerProxy } from 'n8n-workflow';
|
2023-06-20 10:13:18 -07:00
|
|
|
import { SourceControlGitService } from './sourceControlGit.service.ee';
|
2023-05-31 06:01:57 -07:00
|
|
|
import { UserSettings } from 'n8n-core';
|
2023-06-20 10:13:18 -07:00
|
|
|
import type { PushResult, StatusResult } from 'simple-git';
|
2023-05-31 06:01:57 -07:00
|
|
|
import type { ExportResult } from './types/exportResult';
|
2023-06-20 10:13:18 -07:00
|
|
|
import { SourceControlExportService } from './sourceControlExport.service.ee';
|
2023-05-31 06:01:57 -07:00
|
|
|
import { BadRequestError } from '../../ResponseHelper';
|
|
|
|
import type { ImportResult } from './types/importResult';
|
2023-06-20 10:13:18 -07:00
|
|
|
import type { SourceControlPushWorkFolder } from './types/sourceControlPushWorkFolder';
|
|
|
|
import type { SourceControllPullOptions } from './types/sourceControlPullWorkFolder';
|
2023-05-31 06:01:57 -07:00
|
|
|
import type {
|
2023-06-20 10:13:18 -07:00
|
|
|
SourceControlledFileLocation,
|
|
|
|
SourceControlledFile,
|
|
|
|
SourceControlledFileStatus,
|
|
|
|
SourceControlledFileType,
|
|
|
|
} from './types/sourceControlledFile';
|
|
|
|
import { SourceControlPreferencesService } from './sourceControlPreferences.service.ee';
|
2023-05-31 06:01:57 -07:00
|
|
|
import { writeFileSync } from 'fs';
|
2023-06-20 10:13:18 -07:00
|
|
|
import { SourceControlImportService } from './sourceControlImport.service.ee';
|
2023-06-28 02:06:40 -07:00
|
|
|
import type { WorkflowEntity } from '../../databases/entities/WorkflowEntity';
|
|
|
|
import type { CredentialsEntity } from '../../databases/entities/CredentialsEntity';
|
2023-04-19 08:46:10 -07:00
|
|
|
@Service()
|
2023-06-20 10:13:18 -07:00
|
|
|
export class SourceControlService {
|
2023-05-31 06:01:57 -07:00
|
|
|
private sshKeyName: string;
|
|
|
|
|
|
|
|
private sshFolder: string;
|
|
|
|
|
|
|
|
private gitFolder: string;
|
|
|
|
|
|
|
|
constructor(
|
2023-06-20 10:13:18 -07:00
|
|
|
private gitService: SourceControlGitService,
|
|
|
|
private sourceControlPreferencesService: SourceControlPreferencesService,
|
|
|
|
private sourceControlExportService: SourceControlExportService,
|
|
|
|
private sourceControlImportService: SourceControlImportService,
|
2023-05-31 06:01:57 -07:00
|
|
|
) {
|
|
|
|
const userFolder = UserSettings.getUserN8nFolderPath();
|
2023-06-20 10:13:18 -07:00
|
|
|
this.sshFolder = path.join(userFolder, SOURCE_CONTROL_SSH_FOLDER);
|
|
|
|
this.gitFolder = path.join(userFolder, SOURCE_CONTROL_GIT_FOLDER);
|
|
|
|
this.sshKeyName = path.join(this.sshFolder, SOURCE_CONTROL_SSH_KEY_NAME);
|
2023-05-31 06:01:57 -07:00
|
|
|
}
|
2023-04-19 08:46:10 -07:00
|
|
|
|
|
|
|
async init(): Promise<void> {
|
2023-05-31 06:01:57 -07:00
|
|
|
this.gitService.resetService();
|
2023-06-20 10:13:18 -07:00
|
|
|
sourceControlFoldersExistCheck([this.gitFolder, this.sshFolder]);
|
|
|
|
await this.sourceControlPreferencesService.loadFromDbAndApplySourceControlPreferences();
|
2023-05-31 06:01:57 -07:00
|
|
|
await this.gitService.initService({
|
2023-06-20 10:13:18 -07:00
|
|
|
sourceControlPreferences: this.sourceControlPreferencesService.getPreferences(),
|
2023-05-31 06:01:57 -07:00
|
|
|
gitFolder: this.gitFolder,
|
|
|
|
sshKeyName: this.sshKeyName,
|
|
|
|
sshFolder: this.sshFolder,
|
|
|
|
});
|
2023-04-19 08:46:10 -07:00
|
|
|
}
|
|
|
|
|
2023-05-31 06:01:57 -07:00
|
|
|
async disconnect(options: { keepKeyPair?: boolean } = {}) {
|
|
|
|
try {
|
2023-06-20 10:13:18 -07:00
|
|
|
await this.sourceControlPreferencesService.setPreferences({
|
2023-05-31 06:01:57 -07:00
|
|
|
connected: false,
|
|
|
|
branchName: '',
|
|
|
|
});
|
2023-06-20 10:13:18 -07:00
|
|
|
await this.sourceControlExportService.deleteRepositoryFolder();
|
2023-05-31 06:01:57 -07:00
|
|
|
if (!options.keepKeyPair) {
|
2023-06-20 10:13:18 -07:00
|
|
|
await this.sourceControlPreferencesService.deleteKeyPairFiles();
|
2023-05-31 06:01:57 -07:00
|
|
|
}
|
|
|
|
this.gitService.resetService();
|
2023-06-20 10:13:18 -07:00
|
|
|
return this.sourceControlPreferencesService.sourceControlPreferences;
|
2023-05-31 06:01:57 -07:00
|
|
|
} catch (error) {
|
2023-06-20 10:13:18 -07:00
|
|
|
throw Error(`Failed to disconnect from source control: ${(error as Error).message}`);
|
2023-05-31 06:01:57 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-20 10:13:18 -07:00
|
|
|
async initializeRepository(preferences: SourceControlPreferences) {
|
2023-05-31 06:01:57 -07:00
|
|
|
if (!this.gitService.git) {
|
|
|
|
await this.init();
|
|
|
|
}
|
|
|
|
LoggerProxy.debug('Initializing repository...');
|
|
|
|
await this.gitService.initRepository(preferences);
|
|
|
|
let getBranchesResult;
|
|
|
|
try {
|
|
|
|
getBranchesResult = await this.getBranches();
|
|
|
|
} catch (error) {
|
|
|
|
if ((error as Error).message.includes('Warning: Permanently added')) {
|
|
|
|
LoggerProxy.debug('Added repository host to the list of known hosts. Retrying...');
|
|
|
|
getBranchesResult = await this.getBranches();
|
|
|
|
} else {
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (getBranchesResult.branches.includes(preferences.branchName)) {
|
|
|
|
await this.gitService.setBranch(preferences.branchName);
|
|
|
|
} else {
|
|
|
|
if (getBranchesResult.branches?.length === 0) {
|
|
|
|
try {
|
2023-06-20 10:13:18 -07:00
|
|
|
writeFileSync(path.join(this.gitFolder, '/README.md'), SOURCE_CONTROL_README);
|
2023-05-31 06:01:57 -07:00
|
|
|
|
|
|
|
await this.gitService.stage(new Set<string>(['README.md']));
|
|
|
|
await this.gitService.commit('Initial commit');
|
|
|
|
await this.gitService.push({
|
|
|
|
branch: preferences.branchName,
|
|
|
|
force: true,
|
|
|
|
});
|
|
|
|
getBranchesResult = await this.getBranches();
|
|
|
|
} catch (fileError) {
|
|
|
|
LoggerProxy.error(`Failed to create initial commit: ${(fileError as Error).message}`);
|
|
|
|
}
|
|
|
|
} else {
|
2023-06-20 10:13:18 -07:00
|
|
|
await this.sourceControlPreferencesService.setPreferences({
|
2023-05-31 06:01:57 -07:00
|
|
|
branchName: '',
|
|
|
|
connected: true,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return getBranchesResult;
|
2023-04-19 08:46:10 -07:00
|
|
|
}
|
|
|
|
|
2023-05-31 06:01:57 -07:00
|
|
|
async export() {
|
|
|
|
const result: {
|
|
|
|
tags: ExportResult | undefined;
|
|
|
|
credentials: ExportResult | undefined;
|
|
|
|
variables: ExportResult | undefined;
|
|
|
|
workflows: ExportResult | undefined;
|
|
|
|
} = {
|
|
|
|
credentials: undefined,
|
|
|
|
tags: undefined,
|
|
|
|
variables: undefined,
|
|
|
|
workflows: undefined,
|
2023-04-24 08:13:25 -07:00
|
|
|
};
|
2023-05-31 06:01:57 -07:00
|
|
|
try {
|
|
|
|
// comment next line if needed
|
2023-06-20 10:13:18 -07:00
|
|
|
await this.sourceControlExportService.cleanWorkFolder();
|
|
|
|
result.tags = await this.sourceControlExportService.exportTagsToWorkFolder();
|
|
|
|
result.variables = await this.sourceControlExportService.exportVariablesToWorkFolder();
|
|
|
|
result.workflows = await this.sourceControlExportService.exportWorkflowsToWorkFolder();
|
|
|
|
result.credentials = await this.sourceControlExportService.exportCredentialsToWorkFolder();
|
2023-05-31 06:01:57 -07:00
|
|
|
} catch (error) {
|
|
|
|
throw new BadRequestError((error as { message: string }).message);
|
|
|
|
}
|
|
|
|
return result;
|
2023-04-24 08:13:25 -07:00
|
|
|
}
|
|
|
|
|
2023-06-20 10:13:18 -07:00
|
|
|
async import(options: SourceControllPullOptions): Promise<ImportResult | undefined> {
|
2023-05-31 06:01:57 -07:00
|
|
|
try {
|
2023-06-20 10:13:18 -07:00
|
|
|
return await this.sourceControlImportService.importFromWorkFolder(options);
|
2023-05-31 06:01:57 -07:00
|
|
|
} catch (error) {
|
|
|
|
throw new BadRequestError((error as { message: string }).message);
|
2023-04-19 08:46:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-31 06:01:57 -07:00
|
|
|
async getBranches(): Promise<{ branches: string[]; currentBranch: string }> {
|
|
|
|
// fetch first to get include remote changes
|
|
|
|
await this.gitService.fetch();
|
|
|
|
return this.gitService.getBranches();
|
|
|
|
}
|
|
|
|
|
|
|
|
async setBranch(branch: string): Promise<{ branches: string[]; currentBranch: string }> {
|
2023-06-20 10:13:18 -07:00
|
|
|
await this.sourceControlPreferencesService.setPreferences({
|
2023-05-31 06:01:57 -07:00
|
|
|
branchName: branch,
|
2023-06-06 02:23:53 -07:00
|
|
|
connected: branch?.length > 0,
|
2023-04-24 08:13:25 -07:00
|
|
|
});
|
2023-05-31 06:01:57 -07:00
|
|
|
return this.gitService.setBranch(branch);
|
|
|
|
}
|
|
|
|
|
|
|
|
// will reset the branch to the remote branch and pull
|
|
|
|
// this will discard all local changes
|
2023-06-20 10:13:18 -07:00
|
|
|
async resetWorkfolder(options: SourceControllPullOptions): Promise<ImportResult | undefined> {
|
2023-05-31 06:01:57 -07:00
|
|
|
const currentBranch = await this.gitService.getCurrentBranch();
|
2023-06-20 10:13:18 -07:00
|
|
|
await this.sourceControlExportService.cleanWorkFolder();
|
2023-05-31 06:01:57 -07:00
|
|
|
await this.gitService.resetBranch({
|
|
|
|
hard: true,
|
|
|
|
target: currentBranch.remote,
|
|
|
|
});
|
|
|
|
await this.gitService.pull();
|
|
|
|
if (options.importAfterPull) {
|
|
|
|
return this.import(options);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
async pushWorkfolder(
|
2023-06-20 10:13:18 -07:00
|
|
|
options: SourceControlPushWorkFolder,
|
|
|
|
): Promise<PushResult | SourceControlledFile[]> {
|
|
|
|
if (this.sourceControlPreferencesService.isBranchReadOnly()) {
|
2023-05-31 06:01:57 -07:00
|
|
|
throw new BadRequestError('Cannot push onto read-only branch.');
|
|
|
|
}
|
|
|
|
if (!options.skipDiff) {
|
|
|
|
const diffResult = await this.getStatus();
|
|
|
|
const possibleConflicts = diffResult?.filter((file) => file.conflict);
|
|
|
|
if (possibleConflicts?.length > 0 && options.force !== true) {
|
|
|
|
await this.unstage();
|
|
|
|
return diffResult;
|
2023-04-24 08:13:25 -07:00
|
|
|
}
|
|
|
|
}
|
2023-05-31 06:01:57 -07:00
|
|
|
await this.unstage();
|
|
|
|
await this.stage(options);
|
|
|
|
await this.gitService.commit(options.message ?? 'Updated Workfolder');
|
|
|
|
return this.gitService.push({
|
2023-06-20 10:13:18 -07:00
|
|
|
branch: this.sourceControlPreferencesService.getBranchName(),
|
2023-05-31 06:01:57 -07:00
|
|
|
force: options.force ?? false,
|
|
|
|
});
|
2023-04-19 08:46:10 -07:00
|
|
|
}
|
|
|
|
|
2023-05-31 06:01:57 -07:00
|
|
|
async pullWorkfolder(
|
2023-06-20 10:13:18 -07:00
|
|
|
options: SourceControllPullOptions,
|
2023-05-31 06:01:57 -07:00
|
|
|
): Promise<ImportResult | StatusResult | undefined> {
|
|
|
|
await this.resetWorkfolder({
|
|
|
|
importAfterPull: false,
|
|
|
|
userId: options.userId,
|
|
|
|
force: false,
|
2023-04-19 08:46:10 -07:00
|
|
|
});
|
2023-05-31 06:01:57 -07:00
|
|
|
await this.export(); // refresh workfolder
|
|
|
|
const status = await this.gitService.status();
|
|
|
|
|
|
|
|
if (status.modified.length > 0 && options.force !== true) {
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
await this.resetWorkfolder({ ...options, importAfterPull: false });
|
|
|
|
if (options.importAfterPull) {
|
|
|
|
return this.import(options);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
async stage(
|
2023-06-20 10:13:18 -07:00
|
|
|
options: Pick<SourceControlPushWorkFolder, 'fileNames' | 'credentialIds' | 'workflowIds'>,
|
2023-05-31 06:01:57 -07:00
|
|
|
): Promise<{ staged: string[] } | string> {
|
|
|
|
const { fileNames, credentialIds, workflowIds } = options;
|
|
|
|
const status = await this.gitService.status();
|
|
|
|
let mergedFileNames = new Set<string>();
|
|
|
|
fileNames?.forEach((e) => mergedFileNames.add(e));
|
|
|
|
credentialIds?.forEach((e) =>
|
2023-06-20 10:13:18 -07:00
|
|
|
mergedFileNames.add(this.sourceControlExportService.getCredentialsPath(e)),
|
2023-05-31 06:01:57 -07:00
|
|
|
);
|
|
|
|
workflowIds?.forEach((e) =>
|
2023-06-20 10:13:18 -07:00
|
|
|
mergedFileNames.add(this.sourceControlExportService.getWorkflowPath(e)),
|
2023-05-31 06:01:57 -07:00
|
|
|
);
|
|
|
|
if (mergedFileNames.size === 0) {
|
|
|
|
mergedFileNames = new Set<string>([
|
|
|
|
...status.not_added,
|
|
|
|
...status.created,
|
|
|
|
...status.modified,
|
|
|
|
]);
|
|
|
|
}
|
2023-06-28 02:06:40 -07:00
|
|
|
mergedFileNames.add(this.sourceControlExportService.getOwnersPath());
|
2023-05-31 06:01:57 -07:00
|
|
|
const deletedFiles = new Set<string>(status.deleted);
|
|
|
|
deletedFiles.forEach((e) => mergedFileNames.delete(e));
|
|
|
|
await this.unstage();
|
|
|
|
const stageResult = await this.gitService.stage(mergedFileNames, deletedFiles);
|
|
|
|
if (!stageResult) {
|
|
|
|
const statusResult = await this.gitService.status();
|
|
|
|
return { staged: statusResult.staged };
|
|
|
|
}
|
|
|
|
return stageResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
async unstage(): Promise<StatusResult | string> {
|
|
|
|
const stageResult = await this.gitService.resetBranch();
|
|
|
|
if (!stageResult) {
|
|
|
|
return this.gitService.status();
|
|
|
|
}
|
|
|
|
return stageResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
async status(): Promise<StatusResult> {
|
|
|
|
return this.gitService.status();
|
|
|
|
}
|
|
|
|
|
2023-06-20 10:13:18 -07:00
|
|
|
private async fileNameToSourceControlledFile(
|
2023-05-31 06:01:57 -07:00
|
|
|
fileName: string,
|
2023-06-20 10:13:18 -07:00
|
|
|
location: SourceControlledFileLocation,
|
2023-05-31 06:01:57 -07:00
|
|
|
statusResult: StatusResult,
|
2023-06-20 10:13:18 -07:00
|
|
|
): Promise<SourceControlledFile | undefined> {
|
2023-05-31 06:01:57 -07:00
|
|
|
let id: string | undefined = undefined;
|
|
|
|
let name = '';
|
|
|
|
let conflict = false;
|
2023-06-20 10:13:18 -07:00
|
|
|
let status: SourceControlledFileStatus = 'unknown';
|
|
|
|
let type: SourceControlledFileType = 'file';
|
2023-06-28 02:06:40 -07:00
|
|
|
let updatedAt = '';
|
|
|
|
|
|
|
|
const allWorkflows: Map<string, WorkflowEntity> = new Map();
|
|
|
|
(await Db.collections.Workflow.find({ select: ['id', 'name', 'updatedAt'] })).forEach(
|
|
|
|
(workflow) => {
|
|
|
|
allWorkflows.set(workflow.id, workflow);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
const allCredentials: Map<string, CredentialsEntity> = new Map();
|
|
|
|
(await Db.collections.Credentials.find({ select: ['id', 'name', 'updatedAt'] })).forEach(
|
|
|
|
(credential) => {
|
|
|
|
allCredentials.set(credential.id, credential);
|
|
|
|
},
|
|
|
|
);
|
2023-05-31 06:01:57 -07:00
|
|
|
|
|
|
|
// initialize status from git status result
|
|
|
|
if (statusResult.not_added.find((e) => e === fileName)) status = 'new';
|
|
|
|
else if (statusResult.conflicted.find((e) => e === fileName)) {
|
|
|
|
status = 'conflicted';
|
|
|
|
conflict = true;
|
|
|
|
} else if (statusResult.created.find((e) => e === fileName)) status = 'created';
|
|
|
|
else if (statusResult.deleted.find((e) => e === fileName)) status = 'deleted';
|
|
|
|
else if (statusResult.modified.find((e) => e === fileName)) status = 'modified';
|
|
|
|
|
2023-06-20 10:13:18 -07:00
|
|
|
if (fileName.startsWith(SOURCE_CONTROL_WORKFLOW_EXPORT_FOLDER)) {
|
2023-05-31 06:01:57 -07:00
|
|
|
type = 'workflow';
|
|
|
|
if (status === 'deleted') {
|
|
|
|
id = fileName
|
2023-06-20 10:13:18 -07:00
|
|
|
.replace(SOURCE_CONTROL_WORKFLOW_EXPORT_FOLDER, '')
|
2023-05-31 06:01:57 -07:00
|
|
|
.replace(/[\/,\\]/, '')
|
|
|
|
.replace('.json', '');
|
|
|
|
if (location === 'remote') {
|
2023-06-28 02:06:40 -07:00
|
|
|
const existingWorkflow = allWorkflows.get(id);
|
|
|
|
if (existingWorkflow) {
|
|
|
|
name = existingWorkflow.name;
|
|
|
|
updatedAt = existingWorkflow.updatedAt.toISOString();
|
2023-05-31 06:01:57 -07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
name = '(deleted)';
|
2023-06-28 02:06:40 -07:00
|
|
|
// todo: once we have audit log, this deletion date could be looked up
|
2023-05-31 06:01:57 -07:00
|
|
|
}
|
|
|
|
} else {
|
2023-06-20 10:13:18 -07:00
|
|
|
const workflow = await this.sourceControlExportService.getWorkflowFromFile(fileName);
|
2023-05-31 06:01:57 -07:00
|
|
|
if (!workflow?.id) {
|
|
|
|
if (location === 'local') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
id = fileName
|
2023-06-20 10:13:18 -07:00
|
|
|
.replace(SOURCE_CONTROL_WORKFLOW_EXPORT_FOLDER + '/', '')
|
2023-05-31 06:01:57 -07:00
|
|
|
.replace('.json', '');
|
|
|
|
status = 'created';
|
|
|
|
} else {
|
|
|
|
id = workflow.id;
|
|
|
|
name = workflow.name;
|
2023-04-19 08:46:10 -07:00
|
|
|
}
|
2023-06-28 02:06:40 -07:00
|
|
|
const existingWorkflow = allWorkflows.get(id);
|
|
|
|
if (existingWorkflow) {
|
|
|
|
name = existingWorkflow.name;
|
|
|
|
updatedAt = existingWorkflow.updatedAt.toISOString();
|
|
|
|
}
|
2023-04-24 08:13:25 -07:00
|
|
|
}
|
2023-04-19 08:46:10 -07:00
|
|
|
}
|
2023-06-20 10:13:18 -07:00
|
|
|
if (fileName.startsWith(SOURCE_CONTROL_CREDENTIAL_EXPORT_FOLDER)) {
|
2023-05-31 06:01:57 -07:00
|
|
|
type = 'credential';
|
|
|
|
if (status === 'deleted') {
|
|
|
|
id = fileName
|
2023-06-20 10:13:18 -07:00
|
|
|
.replace(SOURCE_CONTROL_CREDENTIAL_EXPORT_FOLDER, '')
|
2023-05-31 06:01:57 -07:00
|
|
|
.replace(/[\/,\\]/, '')
|
|
|
|
.replace('.json', '');
|
|
|
|
if (location === 'remote') {
|
2023-06-28 02:06:40 -07:00
|
|
|
const existingCredential = allCredentials.get(id);
|
|
|
|
if (existingCredential) {
|
|
|
|
name = existingCredential.name;
|
|
|
|
updatedAt = existingCredential.updatedAt.toISOString();
|
2023-05-31 06:01:57 -07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
name = '(deleted)';
|
|
|
|
}
|
|
|
|
} else {
|
2023-06-20 10:13:18 -07:00
|
|
|
const credential = await this.sourceControlExportService.getCredentialFromFile(fileName);
|
2023-05-31 06:01:57 -07:00
|
|
|
if (!credential?.id) {
|
|
|
|
if (location === 'local') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
id = fileName
|
2023-06-20 10:13:18 -07:00
|
|
|
.replace(SOURCE_CONTROL_CREDENTIAL_EXPORT_FOLDER + '/', '')
|
2023-05-31 06:01:57 -07:00
|
|
|
.replace('.json', '');
|
|
|
|
status = 'created';
|
|
|
|
} else {
|
|
|
|
id = credential.id;
|
|
|
|
name = credential.name;
|
|
|
|
}
|
2023-06-28 02:06:40 -07:00
|
|
|
const existingCredential = allCredentials.get(id);
|
|
|
|
if (existingCredential) {
|
|
|
|
name = existingCredential.name;
|
|
|
|
updatedAt = existingCredential.updatedAt.toISOString();
|
|
|
|
}
|
2023-05-31 06:01:57 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-20 10:13:18 -07:00
|
|
|
if (fileName.startsWith(SOURCE_CONTROL_VARIABLES_EXPORT_FILE)) {
|
2023-05-31 06:01:57 -07:00
|
|
|
id = 'variables';
|
|
|
|
name = 'variables';
|
|
|
|
type = 'variables';
|
|
|
|
}
|
|
|
|
|
2023-06-20 10:13:18 -07:00
|
|
|
if (fileName.startsWith(SOURCE_CONTROL_TAGS_EXPORT_FILE)) {
|
2023-06-28 02:06:40 -07:00
|
|
|
const lastUpdatedTag = await Db.collections.Tag.find({
|
|
|
|
order: { updatedAt: 'DESC' },
|
|
|
|
take: 1,
|
|
|
|
select: ['updatedAt'],
|
|
|
|
});
|
2023-05-31 06:01:57 -07:00
|
|
|
id = 'tags';
|
|
|
|
name = 'tags';
|
|
|
|
type = 'tags';
|
2023-06-28 02:06:40 -07:00
|
|
|
updatedAt = lastUpdatedTag[0]?.updatedAt.toISOString();
|
2023-05-31 06:01:57 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!id) return;
|
|
|
|
|
|
|
|
return {
|
|
|
|
file: fileName,
|
|
|
|
id,
|
|
|
|
name,
|
|
|
|
type,
|
|
|
|
status,
|
|
|
|
location,
|
|
|
|
conflict,
|
2023-06-28 02:06:40 -07:00
|
|
|
updatedAt,
|
2023-05-31 06:01:57 -07:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-06-20 10:13:18 -07:00
|
|
|
async getStatus(): Promise<SourceControlledFile[]> {
|
2023-05-31 06:01:57 -07:00
|
|
|
await this.export();
|
|
|
|
await this.stage({});
|
|
|
|
await this.gitService.fetch();
|
2023-06-20 10:13:18 -07:00
|
|
|
const sourceControlledFiles: SourceControlledFile[] = [];
|
2023-05-31 06:01:57 -07:00
|
|
|
const diffRemote = await this.gitService.diffRemote();
|
|
|
|
const diffLocal = await this.gitService.diffLocal();
|
|
|
|
const status = await this.gitService.status();
|
|
|
|
await Promise.all([
|
|
|
|
...(diffRemote?.files.map(async (e) => {
|
2023-06-20 10:13:18 -07:00
|
|
|
const resolvedFile = await this.fileNameToSourceControlledFile(e.file, 'remote', status);
|
2023-05-31 06:01:57 -07:00
|
|
|
if (resolvedFile) {
|
2023-06-20 10:13:18 -07:00
|
|
|
sourceControlledFiles.push(resolvedFile);
|
2023-05-31 06:01:57 -07:00
|
|
|
}
|
|
|
|
}) ?? []),
|
|
|
|
...(diffLocal?.files.map(async (e) => {
|
2023-06-20 10:13:18 -07:00
|
|
|
const resolvedFile = await this.fileNameToSourceControlledFile(e.file, 'local', status);
|
2023-05-31 06:01:57 -07:00
|
|
|
if (resolvedFile) {
|
2023-06-20 10:13:18 -07:00
|
|
|
sourceControlledFiles.push(resolvedFile);
|
2023-05-31 06:01:57 -07:00
|
|
|
}
|
|
|
|
}) ?? []),
|
|
|
|
]);
|
2023-06-20 10:13:18 -07:00
|
|
|
sourceControlledFiles.forEach((e, index, array) => {
|
2023-05-31 06:01:57 -07:00
|
|
|
const similarItems = array.filter(
|
|
|
|
(f) => f.type === e.type && (f.file === e.file || f.id === e.id),
|
|
|
|
);
|
|
|
|
if (similarItems.length > 1) {
|
|
|
|
similarItems.forEach((item) => {
|
|
|
|
item.conflict = true;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
2023-06-20 10:13:18 -07:00
|
|
|
return sourceControlledFiles;
|
2023-04-19 08:46:10 -07:00
|
|
|
}
|
|
|
|
}
|