2021-08-29 11:58:11 -07:00
|
|
|
/* eslint-disable no-param-reassign */
|
|
|
|
/* eslint-disable @typescript-eslint/no-use-before-define */
|
|
|
|
/* eslint-disable @typescript-eslint/no-unsafe-call */
|
|
|
|
/* eslint-disable @typescript-eslint/no-unsafe-assignment */
|
2022-04-08 14:32:08 -07:00
|
|
|
import fs from 'fs';
|
|
|
|
import path from 'path';
|
2021-10-18 20:57:49 -07:00
|
|
|
import { createHash, randomBytes } from 'crypto';
|
2021-08-29 11:58:11 -07:00
|
|
|
// eslint-disable-next-line import/no-cycle
|
2019-06-23 03:35:23 -07:00
|
|
|
import {
|
|
|
|
ENCRYPTION_KEY_ENV_OVERWRITE,
|
|
|
|
EXTENSIONS_SUBDIRECTORY,
|
2020-10-22 06:46:03 -07:00
|
|
|
IUserSettings,
|
2019-06-23 03:35:23 -07:00
|
|
|
USER_FOLDER_ENV_OVERWRITE,
|
|
|
|
USER_SETTINGS_FILE_NAME,
|
|
|
|
USER_SETTINGS_SUBFOLDER,
|
|
|
|
} from '.';
|
|
|
|
|
2021-08-29 11:58:11 -07:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-var-requires
|
2019-06-23 03:35:23 -07:00
|
|
|
const { promisify } = require('util');
|
2021-08-29 11:58:11 -07:00
|
|
|
|
2019-06-23 03:35:23 -07:00
|
|
|
const fsAccess = promisify(fs.access);
|
|
|
|
const fsReadFile = promisify(fs.readFile);
|
|
|
|
const fsMkdir = promisify(fs.mkdir);
|
|
|
|
const fsWriteFile = promisify(fs.writeFile);
|
|
|
|
|
2021-08-29 11:58:11 -07:00
|
|
|
let settingsCache: IUserSettings | undefined;
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates the user settings if they do not exist yet
|
|
|
|
*
|
|
|
|
* @export
|
|
|
|
*/
|
|
|
|
export async function prepareUserSettings(): Promise<IUserSettings> {
|
|
|
|
const settingsPath = getUserSettingsPath();
|
|
|
|
|
|
|
|
let userSettings = await getUserSettings(settingsPath);
|
|
|
|
if (userSettings !== undefined) {
|
|
|
|
// Settings already exist, check if they contain the encryptionKey
|
|
|
|
if (userSettings.encryptionKey !== undefined) {
|
2021-10-18 20:57:49 -07:00
|
|
|
// Key already exists
|
|
|
|
if (userSettings.instanceId === undefined) {
|
|
|
|
userSettings.instanceId = await generateInstanceId(userSettings.encryptionKey);
|
|
|
|
settingsCache = userSettings;
|
|
|
|
}
|
|
|
|
|
2019-06-23 03:35:23 -07:00
|
|
|
return userSettings;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
userSettings = {};
|
|
|
|
}
|
|
|
|
|
2020-06-30 11:08:52 -07:00
|
|
|
if (process.env[ENCRYPTION_KEY_ENV_OVERWRITE] !== undefined) {
|
|
|
|
// Use the encryption key which got set via environment
|
|
|
|
userSettings.encryptionKey = process.env[ENCRYPTION_KEY_ENV_OVERWRITE];
|
|
|
|
} else {
|
|
|
|
// Generate a new encryption key
|
|
|
|
userSettings.encryptionKey = randomBytes(24).toString('base64');
|
|
|
|
}
|
2019-06-23 03:35:23 -07:00
|
|
|
|
2021-10-18 20:57:49 -07:00
|
|
|
userSettings.instanceId = await generateInstanceId(userSettings.encryptionKey);
|
|
|
|
|
2021-08-29 11:58:11 -07:00
|
|
|
// eslint-disable-next-line no-console
|
2021-08-27 08:25:54 -07:00
|
|
|
console.log(`UserSettings were generated and saved to: ${settingsPath}`);
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
return writeUserSettings(userSettings, settingsPath);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the encryption key which is used to encrypt
|
|
|
|
* the credentials.
|
|
|
|
*
|
|
|
|
* @export
|
|
|
|
* @returns
|
|
|
|
*/
|
2021-10-18 20:57:49 -07:00
|
|
|
|
|
|
|
export async function getEncryptionKey(): Promise<string | undefined> {
|
2019-06-23 03:35:23 -07:00
|
|
|
if (process.env[ENCRYPTION_KEY_ENV_OVERWRITE] !== undefined) {
|
|
|
|
return process.env[ENCRYPTION_KEY_ENV_OVERWRITE];
|
|
|
|
}
|
|
|
|
|
|
|
|
const userSettings = await getUserSettings();
|
|
|
|
|
|
|
|
if (userSettings === undefined) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (userSettings.encryptionKey === undefined) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
return userSettings.encryptionKey;
|
|
|
|
}
|
|
|
|
|
2021-10-18 20:57:49 -07:00
|
|
|
/**
|
|
|
|
* Returns the instance ID
|
|
|
|
*
|
|
|
|
* @export
|
|
|
|
* @returns
|
|
|
|
*/
|
|
|
|
export async function getInstanceId(): Promise<string> {
|
|
|
|
const userSettings = await getUserSettings();
|
|
|
|
|
|
|
|
if (userSettings === undefined) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (userSettings.instanceId === undefined) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
return userSettings.instanceId;
|
|
|
|
}
|
|
|
|
|
|
|
|
async function generateInstanceId(key?: string) {
|
|
|
|
const hash = key
|
|
|
|
? createHash('sha256')
|
|
|
|
.update(key.slice(Math.round(key.length / 2)))
|
|
|
|
.digest('hex')
|
|
|
|
: undefined;
|
|
|
|
|
|
|
|
return hash;
|
|
|
|
}
|
|
|
|
|
2019-06-23 03:35:23 -07:00
|
|
|
/**
|
|
|
|
* Adds/Overwrite the given settings in the currently
|
|
|
|
* saved user settings
|
|
|
|
*
|
|
|
|
* @export
|
|
|
|
* @param {IUserSettings} addSettings The settings to add/overwrite
|
|
|
|
* @param {string} [settingsPath] Optional settings file path
|
|
|
|
* @returns {Promise<IUserSettings>}
|
|
|
|
*/
|
2021-08-29 11:58:11 -07:00
|
|
|
export async function addToUserSettings(
|
|
|
|
addSettings: IUserSettings,
|
|
|
|
settingsPath?: string,
|
|
|
|
): Promise<IUserSettings> {
|
2019-06-23 03:35:23 -07:00
|
|
|
if (settingsPath === undefined) {
|
|
|
|
settingsPath = getUserSettingsPath();
|
|
|
|
}
|
|
|
|
|
|
|
|
let userSettings = await getUserSettings(settingsPath);
|
|
|
|
|
|
|
|
if (userSettings === undefined) {
|
|
|
|
userSettings = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add the settings
|
|
|
|
Object.assign(userSettings, addSettings);
|
|
|
|
|
|
|
|
return writeUserSettings(userSettings, settingsPath);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Writes a user settings file
|
|
|
|
*
|
|
|
|
* @export
|
|
|
|
* @param {IUserSettings} userSettings The settings to write
|
|
|
|
* @param {string} [settingsPath] Optional settings file path
|
|
|
|
* @returns {Promise<IUserSettings>}
|
|
|
|
*/
|
2021-08-29 11:58:11 -07:00
|
|
|
export async function writeUserSettings(
|
|
|
|
userSettings: IUserSettings,
|
|
|
|
settingsPath?: string,
|
|
|
|
): Promise<IUserSettings> {
|
2019-06-23 03:35:23 -07:00
|
|
|
if (settingsPath === undefined) {
|
|
|
|
settingsPath = getUserSettingsPath();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (userSettings === undefined) {
|
|
|
|
userSettings = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if parent folder exists if not create it.
|
|
|
|
try {
|
|
|
|
await fsAccess(path.dirname(settingsPath));
|
|
|
|
} catch (error) {
|
|
|
|
// Parent folder does not exist so create
|
|
|
|
await fsMkdir(path.dirname(settingsPath));
|
|
|
|
}
|
|
|
|
|
2021-10-18 20:57:49 -07:00
|
|
|
const settingsToWrite = { ...userSettings };
|
|
|
|
if (settingsToWrite.instanceId !== undefined) {
|
|
|
|
delete settingsToWrite.instanceId;
|
|
|
|
}
|
|
|
|
|
|
|
|
await fsWriteFile(settingsPath, JSON.stringify(settingsToWrite, null, '\t'));
|
2019-06-23 03:35:23 -07:00
|
|
|
settingsCache = JSON.parse(JSON.stringify(userSettings));
|
|
|
|
|
|
|
|
return userSettings;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the content of the user settings
|
|
|
|
*
|
|
|
|
* @export
|
|
|
|
* @returns {UserSettings}
|
|
|
|
*/
|
2021-08-29 11:58:11 -07:00
|
|
|
export async function getUserSettings(
|
|
|
|
settingsPath?: string,
|
|
|
|
ignoreCache?: boolean,
|
|
|
|
): Promise<IUserSettings | undefined> {
|
2019-06-23 03:35:23 -07:00
|
|
|
if (settingsCache !== undefined && ignoreCache !== true) {
|
|
|
|
return settingsCache;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (settingsPath === undefined) {
|
|
|
|
settingsPath = getUserSettingsPath();
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
await fsAccess(settingsPath);
|
|
|
|
} catch (error) {
|
|
|
|
// The file does not exist
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
const settingsFile = await fsReadFile(settingsPath, 'utf8');
|
|
|
|
|
2019-11-10 13:06:11 -08:00
|
|
|
try {
|
|
|
|
settingsCache = JSON.parse(settingsFile);
|
|
|
|
} catch (error) {
|
2021-08-29 11:58:11 -07:00
|
|
|
throw new Error(
|
|
|
|
`Error parsing n8n-config file "${settingsPath}". It does not seem to be valid JSON.`,
|
|
|
|
);
|
2019-11-10 13:06:11 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
return settingsCache as IUserSettings;
|
2019-06-23 03:35:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the path to the user settings
|
|
|
|
*
|
|
|
|
* @export
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
export function getUserSettingsPath(): string {
|
|
|
|
const n8nFolder = getUserN8nFolderPath();
|
|
|
|
|
|
|
|
return path.join(n8nFolder, USER_SETTINGS_FILE_NAME);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retruns the path to the n8n folder in which all n8n
|
|
|
|
* related data gets saved
|
|
|
|
*
|
|
|
|
* @export
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
export function getUserN8nFolderPath(): string {
|
|
|
|
let userFolder;
|
|
|
|
if (process.env[USER_FOLDER_ENV_OVERWRITE] !== undefined) {
|
|
|
|
userFolder = process.env[USER_FOLDER_ENV_OVERWRITE] as string;
|
|
|
|
} else {
|
|
|
|
userFolder = getUserHome();
|
|
|
|
}
|
|
|
|
|
|
|
|
return path.join(userFolder, USER_SETTINGS_SUBFOLDER);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the path to the n8n user folder with the custom
|
|
|
|
* extensions like nodes and credentials
|
|
|
|
*
|
|
|
|
* @export
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
export function getUserN8nFolderCustomExtensionPath(): string {
|
|
|
|
return path.join(getUserN8nFolderPath(), EXTENSIONS_SUBDIRECTORY);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the home folder path of the user if
|
|
|
|
* none can be found it falls back to the current
|
|
|
|
* working directory
|
|
|
|
*
|
|
|
|
* @export
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
export function getUserHome(): string {
|
|
|
|
let variableName = 'HOME';
|
|
|
|
if (process.platform === 'win32') {
|
|
|
|
variableName = 'USERPROFILE';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (process.env[variableName] === undefined) {
|
|
|
|
// If for some reason the variable does not exist
|
|
|
|
// fall back to current folder
|
|
|
|
return process.cwd();
|
|
|
|
}
|
|
|
|
|
|
|
|
return process.env[variableName] as string;
|
|
|
|
}
|